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