以命名管道方式实现网络进程间通信的一个实例

来源:互联网 发布:西安大数据公司 编辑:程序博客网 时间:2024/05/21 06:21
Win32API集支持的命名管道(named pipe)是实现网络进程间通信的一种有效手段。命名管道的接口比较简单,在程序中实现比较方便,由其开发的应用程序也很容易使用,只要知道机器及管道的名称即可连接两台机器上的两个进程。
命名管道首先由一进程调用CreateNamedPipe函数创建,这一创建命名管道的进程称为Server进程,然后Server进程通过调用ConnectNamedPipe函数等待一客户相连。在管道的另一段,Client进程用CreateFile函数或CallNamedPipe函数打开管道句柄,若无可用的管道实例,Client进程可调用WaitNamedPipe函数等待。一旦连接成功,Client进程和Server进程可调用函数ReadFileWriteFileReadFileExWriteileEx传输信息。
笔者在应用程序开发过程中,曾用命名管道方法做了实现网络两进程间通信的一个小例子。这一例子实现了网络两进程间文本的实时互编辑操作,即在任一台机器上对文本的修改可实时反映到另一台机器上,就好像这一修改是在另一台机器上进行的一样。下面就对这一实验程序作一介绍。
程序分两部分:Server部分和Client部分。文中只介绍与命名管道实现有关的内容。
Server部分
#define BUFSIZE 1280
HANDLE hPipe;
管道句柄
BOOL FConnected;
int Success=0;
typedef struct Infor{DWORD Type;
char Text[BUFSIZE-64];};信息传输结构
struct infor Info;包含编辑控制的对话框
BOOL FAR PASCAL _export PipeServerDlg(HWND hDlg UINT message,WPARAM wParam,LPARAM IParam)
{
UINT IdTimer,DWORD ret,dwMode;
switch(message)
{
case WM_INITDIALOG;
IdTimer = SetTimer(hDlg,IdTimer,50,(TIMERPROC)NULL);Server创建命名管道
hPipe = CreateNamedPipe("////.//pipe//buf",管道名
PIPE_ACCESS_DUPLEX,双向传输
PIPE_WAIT|PIPE_TYPE_MESSAGE,阻塞模式,消息流管道
PIPE_UNLIMITED_INSTANCES,,可以创建数目不限的管道实例
BUFSIZE,BUFSIZE,1000,NULL);
If (hPipe!=INVALID_HANDLE_VALUE)
{管道句柄有效
FConnected = ConnectNamedPipe(hPipe,NULL);等待连接
If (FConected)
Succes=1;连接成功
Else
{
CloseHandle(hipe);
Succes=0;
}
}
if (Succes==1)
{    若连接成功,将阻塞模式变为非阻塞模式;设置读模式为消息模式
dwMode=PIPE_READMODE_MESSAGE|PIPE_NOWAIT;
SetNamedPipeHandleState(hPipe,&dwMode,NULL,NULL);
}
return(TRUE);
case WM_TIMER;
if (Succes==1)
{   若连接成功,定时读管道数据
Info.Type=-1;设初始值
If (ReadFile(hPipe,(char*)&Info,sizeof(Infor),&ret,NULL)==TRUE)
{   读管道数据
if (Info.Type==-2)
{   收到对方结束对话框信号,与对方同时结束对话框
KillTimer(hDig,IdTimer); CloseHandle(hPipe); Succes=0;
EndDialog(hDlg,TRUE); return(TRUE);
}
if (Info.Type==1)
SetDlgItemText(hDlg,IDC_EDIT1,Info.Text);
根据对方的变化修改编辑控制内容
}
}
break;
case WM_COMMAND;
if (GET_WM_COMMAND_ID(wParam,lParam)==IDOK)
{   结束对话框
KillTimer(hDlg.IdTimer);
If (Succes==1)
{
Info.Type = -2;置结束对话框信号
WriteFile(hPipe,(char*)&Info,sizeof(Infor),&ret,NULL);写数据到管道
CloseHandle(hPipe);
}
Succes=0;
EndDialog(hDlg,TRUE);
return(TRUE);
}
if (Succes==1&& LOWORD(wParam)==IDC_EDIT1&& HIWORD(wParam)== EN_UPDATE)
{   得到已修改的编辑控制中的内容,并写入管道
GetDlgItemText(hDlg,IDC_EDIT1,(LPSTR)(Info.Text),BUFSIZE-64);
Info.Type=1;Info.Type=1:修改
WriteFile(hPipe,(char*)&Info,sizeof(Infor),&ret,NULL);
}
break;
}
return(FALSE);
}
Client部分
#define BUFSIZE 1280
HANDLE hPipe管道句柄
BOOL FConnected;
int Succes=0,Flag=0;
 
信息传输结构
typedef struct Infor
{
DWORD Type;
char Text[BUFSIZE-64];
};
struct Infor Info;
包含编辑控制的对话框
BOOL FAR PASCAL __export PipeClientDlg(HWND hDlg,UINT message,WPARAM wParam, LPARAM lParam)
{
UINT IdTimer;
DWORD ret,dwMode;
switch(message)
{
case WM_INITDIALOG:
IdTimer=SetTimer(hDlg,IdTimer,50,(TIMERPROC)NULL);
return(TRUE);
case WM_TIMER;
if (Succes==0&&Flag==0)
{    若未打开管道句柄,定时作打开操作
Client打开管道句柄
hPipe = CreateFile("////ServerName//pipe//buf", 管道名,ServerName为运行服务器进程的机器名称
GENERIC_READ|GENERIC_WRITE, /写模式
0,文件非共享
NULL,OPEN_EXISTING,0,NULL);
If (hPipe!=INVALID_HANDLE_VALUE)
{    句柄有效
Succes=1;
Flag=1;
}
else
{
Flag=1;
If (!WaitNamedPipe("////ServerName//pipe//buf",20000)等待
Flag=0;
Clse
Succes=1;
}
if (Succes==1)
{    若连接成功,将阻塞模式变为非阻塞模式;设置读模式为消息模式
dwMode = PIPE_READMODE_MESSAGE | PIPE_NOWAIT;
SetNamedPipeHandleState(hPipe,&dwMode,NULL,NULL);
}
}
if(Succes==1)
{    若连接成功,定时读管道数据
Info.Type=-1;设置初值
If (ReadFile(hPipe,(char*)&Info,sizeof(Infor),&ret,NULL)==TRUE)
{
if (Info1.Type==-2)
{    收到对方结束对话框信号,与对方同时结束对话框
KillTimer(hDlg,IdTimer);CloseHandle(hPipe);
Succes = 0;
Flag = 0;
EndDialog(hDlg,TRUE);
return(TRUE);
}
if (Info.Type==1)
SetDlgItemText(hDlg.IDC_EDIT1,Info.Text);
}
}
break;
case WM_COMMAND;
if (GET_WM_COMMAND_ID(wParam,1Param)==IDOK)
{    结束对话框
KillTimer(hDlg,IdTimer);
If (Succes==1)
{
Info.Type = -2;
WriteFile(hPipe,(char*)&Info,sizeof(Infor),&ret,NULL);
CloseHandle(hPipe);
}
Succes = 0;
Flag = 0;
EndDialog(hDlg,TRUE);
return(TRUE);
}
if(Succes==1&&LOWORD(wParam)==IDC_EDIT1&&HIWORD(wParam)==EN_UPDATE)
{
GetDlgItemText(hDlg,IDC_ENIT1,(LPSTR)(Info.Text),BUFSIZE-64);
Info.Type = 1;
WriteFile(hPipe,(char*)&Info,sizeof(Infor),&ret,NULL);
}
break;
}
return(FALSE);
}