window32下的進程通訊總結(jié)
1。內(nèi)存映射文件
方法
a。創(chuàng)建 在發(fā)送數(shù)據(jù)的進程中調(diào)用CreateFileMapping創(chuàng)建有名的共享內(nèi)存:
HANDLE
CreateFileMapping(
HANDLE
hFile, // 映射文件的句柄,
//設(shè)為0xFFFFFFFF以創(chuàng)建一個進程間共享的對象
LPSECURITY_ATTRIBUTES
lpFileMappingAttributes, // 安全屬性
DWORD flProtect,
// 保護方式
DWORD
dwMaximumSizeHigh,
//對象的大小
DWORD dwMaximumSizeLow,
LPCTSTR
lpName
// 必須為映射文件命名
);
保護方式可以是PAGE_READONLY或是PAGE_READWRITE。如果多進程都對同一共享內(nèi)存進行寫訪問,則必須保持相互間同步。
映射文件還可以指定PAGE_WRITECOPY標志,可以保證其原始數(shù)據(jù)不會遭到破壞。
如
HANDLE hMySharedMapFile=CreateFileMapping((HANDLE)0xFFFFFFFF), NULL,
PAGE_READWRITE,0,0x1000,"MySharedMem");
b。調(diào)用MapViewOfFile函數(shù)映射到本進程的地址空間內(nèi)。
LPSTR pszMySharedMapView=(LPSTR)MapViewOfFile(hMySharedMapFile,
FILE_MAP_READ|FILE_MAP_WRITE,0,0,0);
C。訪問。其他進程訪問共享對象,獲得對象名并調(diào)用OpenFileMapping函數(shù)。
HANDLE
hMySharedMapFile=OpenFileMapping(FILE_MAP_WRITEFALSE,"MySharedMem");
一旦其他進程獲得映射對象的句柄,可以象創(chuàng)建進程那樣調(diào)用MapViewOfFile函數(shù)來映射對象視圖。用戶可以使用該對象視圖來進行數(shù)據(jù)讀寫操作,以達到數(shù)據(jù)通訊的目的。
d。結(jié)束使用共享內(nèi)存后,調(diào)用UnmapViewOfFile函數(shù)以取消其地址空間內(nèi)的視圖:
if
(!UnmapViewOfFile(pszMySharedMapView))
{ AfxMessageBox("could not unmap
view of file"); }
2。共享內(nèi)存DLL
在VC中使用data_seg
pragma宏。如下
#pragma data_seg("MYSEC")
char
MySharedData[4096]={0};
//注意初始化
#pragma data_seg()
然后在用戶的DEF文件中為有名的數(shù)據(jù)區(qū)設(shè)定共享屬性。
LIBRARY TEST
DATA READ WRITE
SECTIONS
.MYSEC READ
WRITE SHARED
這樣每個附屬于DLL的進程都將接受到屬于自己的數(shù)據(jù)拷貝,一個進程的數(shù)據(jù)變化并不會反映到其他進程的數(shù)據(jù)中。
在DEF文件中適當?shù)剌敵鰯?shù)據(jù)。以下的DEF文件項說明了如何以常數(shù)變量的形式輸出MySharedData。
EXPORTS
MySharedData CONSTANT
最后在應(yīng)用程序(進程)按外部變量引用共享數(shù)據(jù)。
extern _export"C"{ char * MySharedData[];}
進程中使用該變量應(yīng)注意間接引用。
m_pStatic=(CEdit*)GetDlgItem(IDC_SHARED);
m_pStatic->GetLine(0,*MySharedData,80);
3。WM_COPYDATA
WM_COPYDATA消息主要目的是允許在進程間傳遞只讀數(shù)據(jù)。SDK文檔推薦用戶使用SendMessage函數(shù),接受方在數(shù)據(jù)拷貝完成前不返回,這樣發(fā)送方就不可能刪除和修改數(shù)據(jù):
SendMessage(hwnd,WM_COPYDATA,wParam,lParam);
其中wParam設(shè)置為包含數(shù)據(jù)的窗口的句柄。lParam指向一個COPYDATASTRUCT的結(jié)構(gòu):
typedef struct
tagCOPYDATASTRUCT{
DWORD dwData; //用戶定義數(shù)據(jù)
DWORD
cbData; //數(shù)據(jù)大小
PVOID
lpData; //指向數(shù)據(jù)的指針
}COPYDATASTRUCT;
在實際應(yīng)用中進程之間需要發(fā)送和接收Windows消息來通知進程間相互通訊,發(fā)送方發(fā)送通訊的消息以通知接收方,接收方在收到發(fā)送方的消息后就可以對內(nèi)存進行讀寫操作。
可以在程序設(shè)計中采用Windows注冊消息進行消息傳遞,首先在發(fā)送進程初始化過程中進行消息注冊:
m_nMsgMapped=::RegisterWindowsMessage("Mapped");
m_nMsgHandle=::RegisterWindowsMessage("Handle");
m_nMsgShared=::RegisterWindowsMessage("Shared");
在程序運行中向接收進程發(fā)送消息:
CWnd* pWndRecv=FindWindow(lpClassName,"Receive");
pWndRecv->SendMessage(m_MsgMapped,0,0);
pWndRecv->SendMessage(m_nMsgHandle,
(UINT)GetCurrentProcessID(),
(LONG)pApp->m_hGlobalHandle);
pWndRecv->SendMessage(m_nMsgShared,0,0);
可以按如下方式發(fā)送WM_COPYDATA消息:
static COPYDATASTRUCT
cds;
//用戶存放數(shù)據(jù)
pWnd->SendMessage(WM_COPYDATA,NULL,(LONG)&cds);
接收方進程初始化也必須進行消息注冊:
UNIT CRecvApp::
m_nMsgMapped=::RegisterWindowsMessage("Mapped");
UNIT
CRecvApp::m_nMsgHandle=::RegisterWindowsMessage("Handle");
UNIT
CRecvApp::m_nMsgShared=::RegisterWindowsMessage("Shared");
同時映射消息函數(shù)如下:
ON_REGISTERED_MASSAGE(CRecvApp::m_nMsgMapped,OnRegMsgMapped)
ON_REGISTERED_MASSAGE(CRecvApp::m_nMsgHandle,OnRegMsgHandle)
ON_REGISTERED_MASSAGE(CRecvApp::m_nMsgShared,OnRegMsgShared)
在這些消息函數(shù)我們就可以采用上述技術(shù)實現(xiàn)接收進程中數(shù)據(jù)的讀寫操作了。
4。調(diào)用ReadProcessMemory以及WriteProcessMemory函數(shù).
調(diào)用ReadProcessMemory以及WriteProcessMemory函數(shù)用戶在發(fā)送進程中分配一塊內(nèi)存存放數(shù)據(jù),調(diào)用GlobalAlloc或VirtualAlloc函數(shù)實現(xiàn):
pApp->m_hGlobalHandle=GlobalAlloc(GMEM_SHARE,1024);
可以得到指針地址:
pApp->mpszGlobalHandlePtr=(LPSTR)GlobalLock
(pApp->m_hGlobalHandle);
在接收進程中要用到用戶希望影響的進程的打開句柄。為了讀寫另一進程,按如下方式調(diào)用OpenProcess函數(shù):
HANDLE
hTargetProcess=OpenProcess(
STANDARD_RIGHTS_REQUIRED|
PROCESS_VM_REDA|
PROCESS_VM_WRITE|
PROCESS_VM_OPERATION,//訪問權(quán)限
FALSE,//繼承關(guān)系
dwProcessID);//進程ID
為保證OpenProcess函數(shù)調(diào)用成功,用戶所影響的進程必須由上述標志創(chuàng)建。
用戶獲得一個進程的有效句柄,就可調(diào)用ReadProcessMemory函數(shù)讀取該進程的內(nèi)存:
BOOL ReadProcessMemory(
HANDLE hProcess, // 進程指針
LPCVOID lpBaseAddress, // 數(shù)據(jù)塊的首地址
LPVOID lpBuffer, // 讀取數(shù)據(jù)所需緩沖區(qū)
DWORD
cbRead, //
要讀取的字節(jié)數(shù)
LPDWORD
lpNumberOfBytesRead
);
使用同樣的句柄也可以寫入該進程的內(nèi)存:
BOOL WriteProcessMemory(
HANDLE
hProcess, // 進程指針
LPVOID
lpBaseAddress, // 要寫入的首地址
LPVOID
lpBuffer, // 緩沖區(qū)地址
DWORD
cbWrite, // 要寫的字節(jié)數(shù)
LPDWORD
lpNumberOfBytesWritten
);