• <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年8月>
            272829303112
            3456789
            10111213141516
            17181920212223
            24252627282930
            31123456

            享受編程

            常用鏈接

            留言簿(11)

            隨筆分類(159)

            隨筆檔案(224)

            文章分類(2)

            文章檔案(4)

            經(jīng)典c++博客

            搜索

            •  

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

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

            最近在寫個(gè)程序的時(shí)候需要在進(jìn)程間通訊,具體需求是這樣。

            1.主要有兩個(gè)進(jìn)程:一個(gè)進(jìn)程作為被請(qǐng)求進(jìn)程,我們稱為 SERVER 進(jìn)程;另一個(gè)進(jìn)程是請(qǐng)求進(jìn)程,稱為 CLIENG 進(jìn)程。

            2.SERVER 進(jìn)程提供一些服務(wù),其完成計(jì)算功能;而 CLIENT 進(jìn)程需要在它執(zhí)行完計(jì)算之后將結(jié)果取會(huì)。

            由于計(jì)算結(jié)果可能是一個(gè)結(jié)構(gòu),也可能是一個(gè)復(fù)雜的數(shù)據(jù),所以通過(guò)消息來(lái)在進(jìn)程傳遞信息是有限的。
            另一方面一般是單方向的通訊,實(shí)際上這里的需求有一個(gè)雙向性,看下圖:

            進(jìn)程間通訊-WriteProcessMemory和ReadProcessMemory - timke - Yes,It is!

            這里兩個(gè)進(jìn)程都可以有自己的窗口,因此實(shí)際上我們可以通過(guò)消息來(lái)通知對(duì)方。但仔細(xì)一想,請(qǐng)求服務(wù)通過(guò)windows消息是沒有問(wèn)題的,
            通知結(jié)果通過(guò)消息是不妥當(dāng)?shù)模瑢?shí)際上我們需要在請(qǐng)求服務(wù)完以后立即得到執(zhí)行結(jié)果,而使用windows消息可能有時(shí)間上的問(wèn)題,而且同步非常麻煩。

            想要的結(jié)果是在 CLIENT 請(qǐng)求服務(wù)以后立即得到返回結(jié)果,但我們可以改變一下思路就容易多了:結(jié)果不是有 SERVER 返回,而由 CLIENG 自己去獲取。
            這樣,我們可以在請(qǐng)求消息的時(shí)候使用 SendMessage 來(lái)發(fā)送這個(gè)請(qǐng)求。這是必須的, SendMessage 發(fā)送消息是同步的方式,必須等到消息處理完畢之后才返回,
            而這正是我們想要的結(jié)果。那么 CLIENT 怎么樣才能獲得 SERVER 進(jìn)程的結(jié)果來(lái)。


            我們知道, WINDOWS 下每個(gè)進(jìn)程都有自己的地址空間,一般情況下一個(gè)進(jìn)程訪問(wèn)你一個(gè)進(jìn)程的地址是不正確的,最簡(jiǎn)單的是提示該地址無(wú)效,
            程序崩潰。當(dāng)然還是有很多中方式來(lái)讀取對(duì)方進(jìn)程的地址空間上的數(shù)據(jù)。

            1.可以做一個(gè) DLL ,利用注入 DLL 的方式,來(lái)將該 DLL 注入到遠(yuǎn)程進(jìn)程的地址空間上,然后通過(guò) DLL 的 API 來(lái)讀取。這個(gè)方式有點(diǎn)麻煩,還必須寫一個(gè) DLL ,還要注入 DLL 。

            2.使用 WriteProcessMemory 和 ReadProcessMemory 來(lái)讀寫遠(yuǎn)程進(jìn)程的內(nèi)存。這種方式相對(duì)比較簡(jiǎn)單,
            其有一個(gè)參數(shù)遠(yuǎn)程進(jìn)程的 HANDLE 指示你需要讀寫哪個(gè)進(jìn)程的內(nèi)存。當(dāng)然,使用這兩個(gè)函數(shù)是需要遠(yuǎn)程進(jìn)程的地址空間的,這個(gè)地址是通過(guò) VirtualAllocEx 來(lái)分配的,
            其通過(guò) VirtualFreeEx 來(lái)釋放。這兩個(gè) API 也有一個(gè)進(jìn)程句柄參數(shù)。

            有人問(wèn),為什么不使用 WM_COPYDATA 來(lái)傳遞大塊數(shù)據(jù)?好,該消息是可以傳遞大塊數(shù)據(jù),但我們的應(yīng)用中需要消息的雙向,所以這里使用 WM_COPYDATA 是不合適的。

             

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

            1.  找到對(duì)方進(jìn)程的窗口。

            2.  找到他的進(jìn)程 ID

            3.  使用 PROCESS_VM_OPERATION| PROCESS_VM_WRITE|PROCESS_VM_READ 標(biāo)志來(lái)打開該進(jìn)程,得到該繼承的 HANDLE 。

            4.  使用 VirtualAllocEx 來(lái)在該進(jìn)程上分配適當(dāng)大小的內(nèi)存,得到一個(gè)地址,這個(gè)地址是遠(yuǎn)程進(jìn)程的,通過(guò)不同的方式來(lái)修改該地址上的值是無(wú)效的。

            5.  如果需要傳遞一些參數(shù)到遠(yuǎn)程進(jìn)程,我們可以在該內(nèi)存上寫一些內(nèi)容,通過(guò) WriteProcessMemory 來(lái)完成

            6.  使用 SendMessage 來(lái)發(fā)送請(qǐng)求服務(wù)消息,同時(shí)將上面分配的內(nèi)存地址作為參數(shù)傳遞給遠(yuǎn)程進(jìn)程。

            7.  遠(yuǎn)程進(jìn)程得到指定消息后處理該消息,取得參數(shù),計(jì)算結(jié)果,將結(jié)果寫到指定的地址。由于這個(gè)地址是遠(yuǎn)程進(jìn)程自己的地址空間,其操作這塊內(nèi)存的方法沒有什么特別之處。

            8.  SendMessage 消息返回, CLIENT 知道后,從上面的內(nèi)存地址中讀取返回結(jié)果;這里必須使用 ReadProcessMemory 來(lái)讀取。好了,整個(gè)過(guò)程結(jié)束。

            9.  調(diào)用 VritualFreeEx 將上面的內(nèi)存釋放。

            這里需要強(qiáng)調(diào)兩點(diǎn):

            1.  打開進(jìn)程的時(shí)候必須設(shè)置對(duì)虛擬內(nèi)存可操作、可寫、可讀,如果只是可寫,那么 ReadProcessMemory 將讀取不正確。

            2.  必須釋放該內(nèi)存。


            下面是部分程序:

            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 漂漂 閱讀(1837) 評(píng)論(0)  編輯 收藏 引用

            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問(wèn)   Chat2DB   管理


            亚洲精品成人网久久久久久| 久久久久亚洲爆乳少妇无| 欧美一区二区三区久久综| 99久久国产综合精品女同图片| 久久久久久亚洲Av无码精品专口| www.久久热.com| 久久久噜噜噜久久中文字幕色伊伊 | 亚洲AⅤ优女AV综合久久久| 亚洲精品99久久久久中文字幕| 久久精品中文无码资源站| 久久久久国色AV免费看图片| 久久精品国产亚洲av麻豆小说| 日本久久中文字幕| 久久精品国产亚洲沈樵| 亚洲狠狠婷婷综合久久久久| 手机看片久久高清国产日韩| 91久久香蕉国产熟女线看| 午夜人妻久久久久久久久| 久久婷婷五月综合97色直播| 久久ZYZ资源站无码中文动漫| 超级97碰碰碰碰久久久久最新| 国产日韩欧美久久| 青草影院天堂男人久久| 日产精品99久久久久久| 久久久久综合中文字幕| 久久e热在这里只有国产中文精品99| 97久久久久人妻精品专区| 久久久久99精品成人片试看| 久久亚洲精品国产精品| 亚洲伊人久久大香线蕉综合图片| 久久午夜福利无码1000合集 | 亚洲?V乱码久久精品蜜桃| 国产精品久久亚洲不卡动漫| 国产高潮国产高潮久久久| 亚洲国产精品无码久久98| 中文国产成人精品久久亚洲精品AⅤ无码精品| 日韩欧美亚洲综合久久影院d3| 超级碰久久免费公开视频| 国产成人久久精品麻豆一区| 精品久久久久久无码人妻热| 欧美国产精品久久高清|