• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            隨筆 - 224  文章 - 41  trackbacks - 0
            <2025年5月>
            27282930123
            45678910
            11121314151617
            18192021222324
            25262728293031
            1234567

            享受編程

            常用鏈接

            留言簿(11)

            隨筆分類(159)

            隨筆檔案(224)

            文章分類(2)

            文章檔案(4)

            經典c++博客

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

            原文地址:http://timke.blog.163.com/blog/static/10158730620103124547256/

            最近在寫個程序的時候需要在進程間通訊,具體需求是這樣。

            1.主要有兩個進程:一個進程作為被請求進程,我們稱為 SERVER 進程;另一個進程是請求進程,稱為 CLIENG 進程。

            2.SERVER 進程提供一些服務,其完成計算功能;而 CLIENT 進程需要在它執行完計算之后將結果取會。

            由于計算結果可能是一個結構,也可能是一個復雜的數據,所以通過消息來在進程傳遞信息是有限的。
            另一方面一般是單方向的通訊,實際上這里的需求有一個雙向性,看下圖:

            進程間通訊-WriteProcessMemory和ReadProcessMemory - timke - Yes,It is!

            這里兩個進程都可以有自己的窗口,因此實際上我們可以通過消息來通知對方。但仔細一想,請求服務通過windows消息是沒有問題的,
            通知結果通過消息是不妥當的,實際上我們需要在請求服務完以后立即得到執行結果,而使用windows消息可能有時間上的問題,而且同步非常麻煩。

            想要的結果是在 CLIENT 請求服務以后立即得到返回結果,但我們可以改變一下思路就容易多了:結果不是有 SERVER 返回,而由 CLIENG 自己去獲取。
            這樣,我們可以在請求消息的時候使用 SendMessage 來發送這個請求。這是必須的, SendMessage 發送消息是同步的方式,必須等到消息處理完畢之后才返回,
            而這正是我們想要的結果。那么 CLIENT 怎么樣才能獲得 SERVER 進程的結果來。


            我們知道, WINDOWS 下每個進程都有自己的地址空間,一般情況下一個進程訪問你一個進程的地址是不正確的,最簡單的是提示該地址無效,
            程序崩潰。當然還是有很多中方式來讀取對方進程的地址空間上的數據。

            1.可以做一個 DLL ,利用注入 DLL 的方式,來將該 DLL 注入到遠程進程的地址空間上,然后通過 DLL 的 API 來讀取。這個方式有點麻煩,還必須寫一個 DLL ,還要注入 DLL 。

            2.使用 WriteProcessMemory 和 ReadProcessMemory 來讀寫遠程進程的內存。這種方式相對比較簡單,
            其有一個參數遠程進程的 HANDLE 指示你需要讀寫哪個進程的內存。當然,使用這兩個函數是需要遠程進程的地址空間的,這個地址是通過 VirtualAllocEx 來分配的,
            其通過 VirtualFreeEx 來釋放。這兩個 API 也有一個進程句柄參數。

            有人問,為什么不使用 WM_COPYDATA 來傳遞大塊數據?好,該消息是可以傳遞大塊數據,但我們的應用中需要消息的雙向,所以這里使用 WM_COPYDATA 是不合適的。

             

            下面我們看一下具體步驟:

            1.  找到對方進程的窗口。

            2.  找到他的進程 ID

            3.  使用 PROCESS_VM_OPERATION| PROCESS_VM_WRITE|PROCESS_VM_READ 標志來打開該進程,得到該繼承的 HANDLE 。

            4.  使用 VirtualAllocEx 來在該進程上分配適當大小的內存,得到一個地址,這個地址是遠程進程的,通過不同的方式來修改該地址上的值是無效的。

            5.  如果需要傳遞一些參數到遠程進程,我們可以在該內存上寫一些內容,通過 WriteProcessMemory 來完成

            6.  使用 SendMessage 來發送請求服務消息,同時將上面分配的內存地址作為參數傳遞給遠程進程。

            7.  遠程進程得到指定消息后處理該消息,取得參數,計算結果,將結果寫到指定的地址。由于這個地址是遠程進程自己的地址空間,其操作這塊內存的方法沒有什么特別之處。

            8.  SendMessage 消息返回, CLIENT 知道后,從上面的內存地址中讀取返回結果;這里必須使用 ReadProcessMemory 來讀取。好了,整個過程結束。

            9.  調用 VritualFreeEx 將上面的內存釋放。

            這里需要強調兩點:

            1.  打開進程的時候必須設置對虛擬內存可操作、可寫、可讀,如果只是可寫,那么 ReadProcessMemory 將讀取不正確。

            2.  必須釋放該內存。


            下面是部分程序:

            CLIENT 程序:

             

            HANDLE hProcess = NULL;
                DWORD dwProcessId 
            = 0;
                HWND hServerWnd 
            = ::FindWindow(NULL,"CompareServer");
                
            if(hServerWnd == NULL)
                {
                    
            //Need create the process
                    return ;
                }
                ::GetWindowThreadProcessId(hServerWnd,
            &dwProcessId);
                hProcess 
            = OpenProcess(PROCESS_VM_OPERATION|
                        PROCESS_VM_WRITE
            |PROCESS_VM_READ,FALSE,dwProcessId);
                
            if(hProcess == NULL) return ;

                MyInfo 
            * pMyInfo = NULL;

                pMyInfo 
            = (MyInfo *)VirtualAllocEx(hProcess,NULL,
                    
            sizeof(MyInfo),MEM_COMMIT,PAGE_READWRITE);

                
            if(pMyInfo == NULL) return ;
                MyInfo myInfo;
                myInfo.blue 
            = 20.01;
                myInfo.red 
            = 3333;

                WriteProcessMemory(hProcess,pMyInfo,
            &myInfo,sizeof(MyInfo),NULL);

                MsgStruct msgStruct;
                msgStruct.dwAddrMem 
            = (DWORD)pMyInfo;
                msgStruct.dwMenLen 
            = sizeof(MyInfo);

                COPYDATASTRUCT cps;
                cps.cbData 
            = sizeof(MsgStruct);
                cps.lpData 
            = (LPBYTE) &msgStruct;
                cps.dwData 
            = 0;

                SIZE_T dwRead
            = 0;
                MyInfo myInfo2;
                ::SendMessage(hServerWnd,WM_COPYDATA,(WPARAM)m_hWnd,(LPARAM)
            &cps);
                BOOL bRet 
            = ::ReadProcessMemory(hProcess,pMyInfo,&myInfo2,sizeof(MyInfo),&dwRead);
                
                dwRead 
            = GetLastError();
                m_log.Format(
            "red =%.2f,blue=%.2f",myInfo2.blue,myInfo2.red);
                TRACE(m_log);
                VirtualFreeEx(hProcess,pMyInfo,
            0,MEM_RELEASE);
                UpdateData(FALSE);


            SERVER 程序:

            LRESULT    CMyWindow::OnCompareImage(HWND hWnd,WPARAM wParam,LPARAM lParam)
            {
                
            if(wParam <sizeof(MyInfo)) return -1;
                MyInfo 
            * pMyInfo = (MyInfo *)lParam;

                sprintf(m_strLog,
            "client:red=%.2f,blue=%.2f",pMyInfo->red,pMyInfo->blue);
                ::TextOut(GetDC(hWnd),
            0,50,m_strLog,strlen(m_strLog));
                pMyInfo
            ->blue = 1.0;
                pMyInfo
            ->red = 2.0;
                
            return 0;
            }

            LRESULT    CMyWindow::OnCopyData(HWND hWnd,WPARAM wParam,LPARAM lParam)
            {
                PCOPYDATASTRUCT    lpcds;
                lpcds 
            = (PCOPYDATASTRUCT)lParam;
                MsgStruct msgStruct;
                memcpy(
            &msgStruct,lpcds->lpData,lpcds->cbData);

                
            if(msgStruct.dwMenLen<sizeof(MyInfo))
                    
            return 0;
                MyInfo 
            * pMyInfo = (MyInfo *)msgStruct.dwAddrMem;
                
            char temp[100];
                sprintf(temp,
            "client:red=%.2f,blue=%.2f",pMyInfo->red,pMyInfo->blue);
                ::TextOut(GetDC(hWnd),
            0,50,temp,strlen(temp));
                pMyInfo
            ->blue = 1.0;
                pMyInfo
            ->red = 2.0;


                
            return 0;
            }




            posted on 2013-07-11 14:23 漂漂 閱讀(1817) 評論(0)  編輯 收藏 引用
            亚洲国产成人精品女人久久久| 久久99久久99精品免视看动漫| 91久久精品无码一区二区毛片| 青青青国产成人久久111网站| 欧美精品福利视频一区二区三区久久久精品 | 亚洲国产成人久久精品99| 国产香蕉久久精品综合网| 亚洲欧美伊人久久综合一区二区 | 久久播电影网| 欧美亚洲色综久久精品国产| 香蕉久久一区二区不卡无毒影院| 亚洲人AV永久一区二区三区久久| 69SEX久久精品国产麻豆| 日日狠狠久久偷偷色综合免费| 久久人人爽人人爽人人片av高请| 久久se精品一区二区影院| 久久精品毛片免费观看| 色老头网站久久网| 久久人妻少妇嫩草AV无码蜜桃| 久久99国产精品二区不卡| 2021国产精品久久精品| 一级做a爰片久久毛片16| 91精品国产高清91久久久久久| 久久精品国产色蜜蜜麻豆| 日韩精品国产自在久久现线拍| 色妞色综合久久夜夜| 亚洲午夜久久久久久噜噜噜| 午夜精品久久影院蜜桃| 久久婷婷色综合一区二区| 99久久精品免费观看国产| 97久久精品人人澡人人爽| 国产一区二区三区久久精品| 久久无码人妻一区二区三区| 国内精品综合久久久40p| 久久精品视频一| 久久精品国产免费观看三人同眠| 久久夜色撩人精品国产小说| 久久精品国产精品亜洲毛片 | 亚洲国产成人久久综合一| 免费观看久久精彩视频| 久久99精品久久久久久|