WM_COPYDATA通信
MFC创建服务端和客户端,客户端想要发送消息给服务端的时候,客户端要拿到服务端的窗口句柄,并且服务端要开启WM_COPYDATA的消息。
客户端发送消息的代码:
void CProcessClientDlg::OnBnClickedButtonSend()
{
//发送数据 COPYDATA
CString strWindowsTitle = "服务端";
CString strMsg = "i am client";
//利用标题拿到句柄
HWND hWnd = ::FindWindow(NULL, strWindowsTitle.GetBuffer(0));
if (hWnd != NULL && IsWindow(hWnd))
{
//数据封装
COPYDATASTRUCT cpd;
cpd.dwData = 0;
cpd.cbData = strMsg.GetLength() * sizeof(TCHAR);
cpd.lpData = (PVOID)strMsg.GetBuffer(0);
::SendMessage(hWnd, WM_COPYDATA, (WPARAM)(AfxGetApp()->m_pMainWnd), (LPARAM)&cpd);
}
strWindowsTitle.ReleaseBuffer();
strMsg.ReleaseBuffer();
}
服务端开启WM_COPYDATA消息。
右键点击服务端的窗口,选择类向导:
服务端接收消息函数代码:
BOOL CProcessCommunitDlg::OnCopyData(CWnd* pWnd, COPYDATASTRUCT* pCopyDataStruct)
{
// TODO: 在此添加消息处理程序代码和/或调用默认值
//copydata 的对应消息函数
LPCTSTR szText = (LPCTSTR)(pCopyDataStruct->lpData);
DWORD dwLength = (DWORD)(pCopyDataStruct->cbData);
TCHAR szRecvText[1024] = { 0 };
memcpy(szRecvText, szText, dwLength);
MessageBox(szRecvText, "Bingo", MB_OK);
return CDialogEx::OnCopyData(pWnd, pCopyDataStruct);
}
注意的是:客户端点击发送按钮后,服务端就自动接收到了消息,并在小窗口显示。
如果服务端想要向客户端发送消息,与上面步骤相反。
总结:
剪贴板比较简单。剪切板和匿名管道只能实现同一机器的两个进程通信而不能实现网络进程之间的通信。
邮槽是基于广播的,可以一对多发送。但只能一个发送,一个接收,要想同时发送接收,须写两次代码。邮槽的缺点传输的数据量很小 424 字节以下。
命名管道和邮槽可以进行网络通信。
命名管道只能是点对点的单一通信。
WM_COPYDATA 封装数据和解析数据,非常方便。如果数据量大,建议用命名管道。