进程间的通信模式:
①、剪贴板
建立一个APPWIZARD,然后创建2个按钮(发送,接收),2个编辑框,
对按钮添加函数;
void CClickDlg::OnButtonSend() { //打开剪贴板,保存信息于剪贴板上 if(OpenClipboard()) { CString str; HANDLE hClip; char *pbuf;// EmptyClipboard(); GetDlgItemText(IDC_EDIT_SEND,str); hClip=GlobalAlloc(GMEM_MOVEABLE,str.GetLength()+1);//分配一个内存对象 //对一个内存对象加锁,并返回其地址; pbuf=(char*)GlobalLock(hClip); strcpy(pbuf,str); GlobalUnlock(hClip); //设置剪贴板的数据 SetClipboardData(CF_TEXT,hClip);//参数格式 CloseClipboard(); } } void CClickDlg::OnButtonRecv() { OpenClipboard(); if (IsClipboardFormatAvailable(CF_TEXT)) { HANDLE hclip; char *pbuf; hclip=GetClipboardData(CF_TEXT);//得到剪贴板的消息 pbuf=(char*)GlobalLock(hclip);//对欲加锁消息加锁 GlobalUnlock(hclip); SetDlgItemText(IDC_EDIT_RECV,pbuf); CloseClipboard(); } }②,通过管道同步消息
创建匿名管道:
1建立单文档appwizard之pipe,添加三个菜单,“创建匿名管道,读取data,写入data”
2添加句柄hread hwrite,
3添加相应菜单“创建匿名管道”的菜单函数
void CPipeView::OnPipeCreate() { SECURITY_ATTRIBUTES sa; sa.bInheritHandle=TRUE; sa.lpSecurityDescriptor=NULL; sa.nLength=sizeof(SECURITY_ATTRIBUTES); if(!CreatePipe(&hRead,&hWrite,&sa,0)) { MessageBox("创建匿名管道失败"); return ; } //启动子进程,交给读写选项; STARTUPINFO stinfo; // memset(&stinfo,0,sizeof(STARTUPINFO)); ZeroMemory(&stinfo,sizeof(STARTUPINFO)); //对所用到的结构成员赋值 stinfo.cb=sizeof(STARTUPINFO); stinfo.dwFlags=STARTF_USESTDHANDLES;//结构体,标准输入\出句柄有效 stinfo.hStdInput=hRead; stinfo.hStdOutput=hWrite;//将进程的标准入\出句柄设置为管道的读写句柄 stinfo.hStdError=GetStdHandle(STD_ERROR_HANDLE);//获得父进程的标准错误句柄; //进程消息的指针LPPROCESS_INFORMATION定义 PROCESS_INFORMATION pl; if(!CreateProcess("E:\\vc6.0\\MSDev98\\MyProjects\\test\\child\\child\\Debug\\child.exe",NULL,NULL,NULL,TRUE,0,NULL,NULL,&stinfo,&pl))//创建一个子线程,实现通过管道进行子进程和父进程的通信; { CloseHandle(hRead); CloseHandle(hWrite); hRead=NULL; hWrite=NULL; MessageBox("创建子进程失败"); return; } else { CloseHandle(pl.hProcess);//将内核对象计数器减 1,为0是收回你内核对象的内存 CloseHandle(pl.hThread); } }重要的是两个函数,一个是Createpipe()和另一个CreateProecess()函数;
3、添加“读取数据”函数响应
void CPipeView::OnPipeRead() { char buf[100]={0}; DWORD dwRead;//保存实际读取的字节数; if(!ReadFile(hRead,buf,100,&dwRead,NULL)) { MessageBox("读取失败"); return; } MessageBox(buf); }4 添加“读取数据”函数响应
void CPipeView::OnPipeWrite() { char buf[]="this is my test pipe"; DWORD dwWrite; if(!WriteFile(hWrite,buf,strlen(buf)+1,&dwWrite,NULL)) { MessageBox("写入失败"); return; } }5编写关于 子进程Child的实现,同样的创建一个appwizard 单文档,命名为“child”,然后添加菜单,“接收data”和“发送data”
6.添加句柄 hread和hwrite 以保存表示输入和输出句柄;
7添加关于View类里面的onnitialUpdate的虚函数,并在这个里面获得标准输入\出句柄;
void CChildView::OnInitialUpdate() { CView::OnInitialUpdate(); //获得标准的读写句柄 hRead=GetStdHandle(STD_INPUT_HANDLE); hWrite=GetStdHandle(STD_OUTPUT_HANDLE); }
8添加Child的菜单的“接收data”和“发送Data”的函数
void CChildView::OnPipeWrite() { char buf[]="匿名管道测试"; DWORD dwWrite; if(!WriteFile(hWrite,buf,strlen(buf)+1,&dwWrite,NULL))//WriteFile()读写管道 { MessageBox("写入失败"); return; } } void CChildView::OnPipeRead() { char buf[100]={0}; DWORD dwRead;//保存实际读取的字节数; if(!ReadFile(hRead,buf,100,&dwRead,NULL))//ReadFile,读管道消息 { MessageBox("读取失败"); return; } MessageBox(buf); }
运行父进程“创建匿名管道”,启动子进程,便可以进行进程之间的通信;
利用命名管道
匿名管道只能在父子子进程进行通信,而.命名管道:还可以跨网络通信,服务器只能在win2000和NT下运行!而客户端可以在95下运行。关键函数CreateNamedPipe
创建一个基于命名管道的进程通信例子
服务器端部分
1.添加一个句柄变量来保存一个命名管道的实例句柄(View类里面)
2.添加“创建命名管道的响应函数”
//创建一个句柄变量来保存一个命名管道的实例句柄 void CNamedPipeView::OnPipreCreate() { hPipe=CreateNamedPipe("\\\\.\\pipe\\mypipe",PIPE_ACCESS_DUPLEX|FILE_FLAG_OVERLAPPED,0,1,1024,1024,0,NULL); if (INVALID_HANDLE_VALUE==hPipe) { MessageBox("创建命名管道失败"); return; } HANDLE hEvent; hEvent=CreateEvent(NULL,TRUE,TRUE,NULL); if (!hEvent) { MessageBox("创建时间失败"); return; } OVERLAPPED owlap; memset(&owlap,0,sizeof(owlap)); owlap.hEvent=hEvent; if(!ConnectNamedPipe(hPipe,&owlap)) { if (ERROR_IO_PENDING!=GetLastError())//此处返回值如果是ERROR_IO_PENDING不表示错误,而是稍后可能继续执行 { MessageBox("等待客户端连接失败!"); CloseHandle(hPipe); CloseHandle(hEvent); hPipe=NULL; return; } if(WAIT_FAILED==WaitForSingleObject(hEvent,INFINITE)) { MessageBox("等待对象失败"); return; } CloseHandle(hEvent); } }3,添加发送和接收部分的函数
void CNamedPipeView::OnEditWrite() { // TODO: Add your command handler code here char buf[]="服务器端写入:匿名管道测试"; DWORD dwWrite; if(!WriteFile(hPipe,buf,strlen(buf)+1,&dwWrite,NULL))//WriteFile()读写管道 { MessageBox("服务器端写入失败"); return; } } void CNamedPipeView::OnEditRead() { char buf[100]={0}; DWORD dwRead;//保存实际读取的字节数; if(!ReadFile(hPipe,buf,100,&dwRead,NULL))//ReadFile,读管道消息 { MessageBox("服务器端读取失败"); return; } MessageBox(buf); }客户端部分的程序:
1,同服务器端部分的第一二部分
2.建立“连接管道的函数”
void CNamedPipeCotView::OnPipeConnect() { // TODO: Add your command handler code here if(!WaitNamedPipe("\\\\.\\pipe\\mypipe",NMPWAIT_WAIT_FOREVER)) { MessageBox("当前没有课命名的管道实例"); return; } hpipe=CreateFile("\\\\.\\pipe\\mypipe",GENERIC_READ|GENERIC_WRITE,0,NULL,OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL,NULL);//可以对文件操作,也可以对管道操作; if (hpipe==INVALID_HANDLE_VALUE) { MessageBox("打开命名管道失败"); hpipe=NULL; return; } }3,建立接收和发送的函数
void CNamedPipeCotView::OnPipeRead() { char buf[100]={0}; DWORD dwRead;//保存实际读取的字节数; if(!ReadFile(hpipe,buf,100,&dwRead,NULL))//ReadFile,读管道消息 { MessageBox("客户端读取失败"); return; } MessageBox(buf); } void CNamedPipeCotView::OnPipeWrite() { char buf[]="客户端写入:匿名管道测试"; DWORD dwWrite; if(!WriteFile(hpipe,buf,strlen(buf)+1,&dwWrite,NULL))//WriteFile()读写管道 { MessageBox("客户端写入失败"); return; } }
四、通过邮槽来实现进程间通信;
邮槽是一种单向传送机制;邮槽是基于广播的,可以一对多发送。但只能一个发送,一个接收,要想同时发送接收,须写两次代码。
服务器端部分:创建邮槽,准备接收数据
void CMailSlotSrvView::OnMailslotRecv() { HANDLE HMailSlot; HMailSlot=CreateMailslot("\\\\.\\mailslot\\MyMailSlot",0,MAILSLOT_WAIT_FOREVER,NULL); if (INVALID_HANDLE_VALUE==HMailSlot) { MessageBox("创建油槽失败"); HMailSlot=NULL; return; } //创建成功,接收数据,因为服务器端只能接收数据; char buf[100]={0}; DWORD dwRead;//保存实际读取的字节数; if(!ReadFile(HMailSlot,buf,100,&dwRead,NULL))//ReadFile,读管道消息 { MessageBox("客户端读取失败"); CloseHandle(HMailSlot); return; } MessageBox(buf); CloseHandle(HMailSlot); }
建立客户端发送数据部分:
void CMailSlotClientView::OnMailslotSend() { HANDLE hMailSlotClient; hMailSlotClient=CreateFile("\\\\.\\mailslot\\MyMailSlot",GENERIC_WRITE,FILE_SHARE_READ,NULL,OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL,NULL);//打开油槽; if (INVALID_HANDLE_VALUE==hMailSlotClient) { MessageBox("打开油槽失败"); hMailSlotClient=NULL; return; } //打开油槽写入数据; char buf[]="客户端写入:匿名管道测试"; DWORD dwWrite; if(!WriteFile(hMailSlotClient,buf,strlen(buf)+1,&dwWrite,NULL))//WriteFile()读写管道 { MessageBox("客户端写入失败"); CloseHandle(hMailSlotClient); return; } CloseHandle(hMailSlotClient); }