第一种:通过自定义消息来实现 SendMessage和PostMessage最大的区别: SendMessage:发送的消息不进入消息队列,而是将指定的消息发送到一个或多个窗口,所以会等待对方的处理,直到对方接收到了才返回 PostMessage:发送的消息进入到消息队列,所以不会等待对方的处理,直接放到消息队列就返回 hmod和dwThreadId参数说明: 如果需要设置局部钩子(作用范围:单一的进程),那么hmod参数为0,dwThreadId设置为需要钩住的线程的ID。 如果需要设置全局钩子(作用范围:所有有效进程),那么hmod参数为模块句柄(钩子的回调函数需要在DLL中),dwThreadId设置为0。includeWindows。hincludecstdioincludeTlHelp32。hintmain(intargc,charargv〔〕){HWNDhwnd;HMODULEhModule;HWNDhWnd;HHOOKgHook;hModuleLoadLibrary(LD:VisualStudioRepos2013进程通信DebugWin32Project1。dll);typedefLRESULT(CALLBACKpWndProc)(intnCode,WPARAMwParam,LPARAMlParam);pWndProcWndProc;WndProc(pWndProc)GetProcAddress(hModule,WndProc12);hWndFindWindow(L32770,LPEDialog);DWORDdwThreadIdGetWindowThreadProcessId(hWnd,NULL);gHookSetWindowsHookEx(WHCALLWNDPROC,WndProc,hModule,dwThreadId);if(NULLgHook){MessageBox(NULL,L安装钩子失败,L提示,MBOKCANCEL);}SendMessage(hWnd,WMUSER100,0,100);PostMessage(hWnd,WMUSER100,0,100);printf(d,GetLastError());return0;} 有个坑:虽然自己没有测试真实性,自己猜想的PostMessage不起作用的原因是对方进程中的消息循环机制中就没有要自己HOOK的循环消息的类型,这时候将该消息放入消息队列当程序本身取出该消息的时候发送给系统,系统通过调用就无法识别要发送给谁,所以说可能如果本身有定义的话应该是可以用PostMessage来实现的,然后说下为什么SendMessage可以实现,因为自己查询了SendMessage该函数是直接将消息发送给窗口的,所以识别的过程就过去省略,直接发送给,以上都当自己放屁发现自己解释的是错的,为什么不删掉?以后当自己看到这里的时候,可以再次回想下,继续思考,重新把它的解释填上!!! DLL编写如下:includestdafx。hincludeTTTT。hincludecstdiopragmacomment(linker,SECTION:。share,RWS)pragmadataseg(。share)HHOOKgHook0;这几句代码申明gHook被多个进程间共享pragmadataseg()externCdeclspec(dllexport)LRESULTCALLBACKWndProc(intnCode,WPARAMwParam,LPARAMlParam){if(nCodeHCACTION){TCHARszWindowText〔〕TEXT(HookTest);PCWPSTRUCTpcw(PCWPSTRUCT)lParam;if(pcwmessageWMUSER100pcwlParam100){MessageBox(0,LWOW,LWOW,0);}}returnCallNextHookEx(gHook,nCode,wParam,lParam);}第二种:通过文件映射创建的共享内存来实现 服务端的代码:includeWindows。hincludecstdiointmain(){HANDLEhMapping;此函数为指定的文件创建一个命名或未命名的文件映射对象hMappingCreateFileMapping(INVALIDHANDLEVALUE,NULL,PAGEREADWRITE,0,0x1000,LMyMapping);if(hMappingNULL){printf(CreateFileMappingFailed,theerrorisd,GetLastError());}LPVOIDpMapMapViewOfFile(hMapping,FILEMAPALLACCESS,0,0,0x1000);if(pMapNULL){printf(MapViewOfFileFailed,theerrorisd,GetLastError());}printf(初始打印:pMap:x,(PDWORD)pMap);getchar();printf(二次打印:pMap:x,(PDWORD)pMap);getchar();UnmapViewOfFile(pMap);CloseHandle(hMapping);} 客户端的代码:includeWindows。hincludecstdiointmain(){HANDLEhMapping;此函数为指定的文件创建一个命名或未命名的文件映射对象hMappingCreateFileMapping(INVALIDHANDLEVALUE,NULL,PAGEREADWRITE,0,0x1000,LMyMapping);if(hMappingNULL){printf(CreateFileMappingFailed,theerrorisd,GetLastError());}LPVOIDpMapMapViewOfFile(hMapping,FILEMAPALLACCESS,0,0,0x1000);if(pMapNULL){printf(MapViewOfFileFailed,theerrorisd,GetLastError());}(PDWORD)pMap0x12345678;printf(初始打印:x,(PDWORD)pMap);getchar();UnmapViewOfFile(pMap);CloseHandle(hMapping);} 第三种方法:匿名管道实现父子进程通信 需要注意的是:父进程创建子进程的时候,关于STARTUPINFO的结构体参数需要比正常多几个如下参数,自己理解来看就是把当前父进程的输入输出流都写到对应的读写管道中si。dwFlagsSTARTFUSESTDHANDLES;si。hStdInputhPipeWrite;si。hStdOutputhPipeRead;si。hStdErrorGetStdHandle(STDERRORHANDLE); 父进程:includeWindows。hincludecstdiointmain(){HANDLEhPipeRead;HANDLEhPipeWrite;SECURITYATTRIBUTESsa;sa。bInheritHandleTRUE;sa。lpSecurityDescriptorNULL;sa。nLengthsizeof(SECURITYATTRIBUTES);BOOLbRetCreatePipe(hPipeRead,hPipeWrite,sa,0);if(!bRet){return1;}STARTUPINFOsi{0};PROCESSINFORMATIONpi;si。cbsizeof(STARTUPINFO);si。dwFlagsSTARTFUSESTDHANDLES;si。hStdInputhPipeWrite;si。hStdOutputhPipeRead;si。hStdErrorGetStdHandle(STDERRORHANDLE);WCHARwszReadBuffer〔MAXPATH〕;WCHARwszWriteBuffer〔MAXPATH〕Lhttp:zpchcbd。com;DWORDdwReadBytes;DWORDdwWriteBytes;写if(WriteFile(hPipeWrite,(LPVOID)wszWriteBuffer,MAXPATH,dwWriteBytes,NULL)){wprintf(LTopProcesss,wszWriteBuffer);}CreateProcess(LD:VisualStudioRepos2013进程通信Debugtest。exe,NULL,NULL,NULL,TRUE,0,NULL,NULL,si,pi);while(true){if(ReadFile(hPipeRead,(LPVOID)wszReadBuffer,MAXPATH,dwReadBytes,NULL)){wprintf(Ls,wszReadBuffer);memset(wszReadBuffer,0,MAXPATH);}}return0;} 子进程:includeWindows。hincludecstdiointmain(){HANDLEhRead;HANDLEhWrite;hReadGetStdHandle(STDOUTPUTHANDLE);hWriteGetStdHandle(STDINPUTHANDLE);WCHARwszBuffer〔MAXPATH〕{0};只读一次DWORDdwReadBytes;if(ReadFile(hRead,wszBuffer,MAXPATH,dwReadBytes,NULL)){MessageBox(0,wszBuffer,L子进程读取到的数据:,MBOK);}else{MessageBox(0,0,L没有读取到数据,MBOK);}return0;} 第四种方法:命名管道进程间通信 服务端:includeWindows。hincludecstdiointmain(){HANDLEhPipeCreateNamedPipe(L。pipeaaaa,PIPEACCESSDUPLEX,PIPEREADMODEBYTEPIPEWAIT,PIPEUNLIMITEDINSTANCES,MAXPATH,MAXPATH,0,NULL);if(hPipeINVALIDHANDLEVALUE){printf(CreatePipeFailed);CloseHandle(hPipe);}WCHARszRecvBuffer〔1024〕;DWORDdwReadBytes;服务端在这里会进行堵塞,等待客户端进行连接if(ConnectNamedPipe(hPipe,NULL)){printf(ConnectNamedPipesuccess);memset(szRecvBuffer,0,1024);if(ReadFile(hPipe,szRecvBuffer,MAXPATH,dwReadBytes,NULL)){wprintf(LReadFileSuccesss,szRecvBuffer);}else{printf(ReadFileFailed,Erroriss,GetLastError());CloseHandle(hPipe);return1;}}CloseHandle(hPipe);return0;} 客户端:includeWindows。hincludecstdiointmain(){HANDLEhPipe;WCHARwszBuffer〔MAXPATH〕Lhahahaha!!!;DWORDdwWriteBytes;if(WaitNamedPipe(L。pipeaaaa,NMPWAITWAITFOREVER)){printf(WaitNamedPipeSuccess);hPipeCreateFile(L。pipeaaaa,GENERICWRITEGENERICREAD,0,NULL,OPENEXISTING,FILEATTRIBUTENORMAL,NULL);if(hPipe!NULL){printf(ConnectPipeSuccess);if(WriteFile(hPipe,wszBuffer,MAXPATH,dwWriteBytes,NULL)){printf(WriteFiledataSuccess);}}}return0;} 第五种方法:DLL共享节 简单的说,其实就是声明DLL中的一段数据区为共享区,在学习DLL共享节的时候找到的如下一段话: 在Win16环境中,DLL的全局数据对每个载入它的进程来说都是相同的,因为所有的进程用的都收同一块地址空间; 而在Win32环境中,情况却发生了变化,每个进程都有了它自己的4GB的内存空间,DLL函数中的代码所创建的任何对象(包括变量)都归调用它的进程所有。 当进程在载入DLL时,操作系统自动把DLL地址映射到该进程的私有空间,也就是进程的虚拟地址空间,而且也复制该DLL的全局数据的一份拷贝到该进程空间。(在物理内存中,多进程载入DLL时,DLL的代码段实际上是只加载了一次,只是将物理地址映射到了各个调用它的进程的虚拟地址空间中,而全局数据会在每个进程都分别加载)。也就是说每个进程所拥有的相同的DLL的全局数据,它们的名称相同,但其值却并不一定是相同的,而且是互不干涉的。 因此,在Win32环境下要想在多个进程中共享数据,就必须进行必要的设置。在访问同一个Dll的各进程之间共享存储器是通过存储器映射文件技术实现的。 那么如何设置呢?方法:也可以把这些需要共享的数据分离出来,放置在一个独立的数据段里,并把该段的属性设置为共享。必须给这些变量赋初值,否则编译器会把没有赋初始值的变量放在一个叫未被初始化的数据段中。 设置的语法格式如下:pragmadataseg(DLLSharedSection)声明共享数据段,并命名该数据段intSharedData123;必须在定义的同时进行初始化!!!!pragmadataseg() 一种方法是在。DEF文件中加入如下语句:SETCTIONSSharedREADWRITESHARED 或者是直接声明: pragmacomment(linker,section:。DLLSharedSection,rws) 或者是在编译器里面声明:在项目设置的链接选项(ProjectSetting〉Link)中加入如下语句: SECTION:Shared,rws dll编写代码:pragmaoncedefineMYAPIexternCdeclspec(dllexport)MYAPIintcdeclgetData();MYAPIvoidcdeclSetData(intyouData);includestdafx。hincludeTTTT。hpragmadataseg(MyShareSection)声明共享数据段,并命名该数据段intSharedData123;必须在定义的同时进行初始化!!!!pragmadataseg()pragmacomment(linker,section:。MyShareSection,rws)MYAPIintcdeclgetData(){returnSharedData;}MYAPIvoidcdeclSetData(intdwData){SharedDatadwData;}