我们先来看看socketpair函数的原型如下:
int socketpair(int domain,int type,int protocol,int sv[])
第一个参数表示协议族,必须为AF_LOCAL;
第二个参数表示类型,既可以是SOCK_STREAM,又可以是SOCK_DGRAM,当参数指定为SOCK_STREAM时,得到的结果称为流管道,它与一般管道的区别是留管道是全双工的,即两个描述符即可读有可写;
第三个参数只能为0;
第四个参数用于保存创建的套接字对;
socketpair函数建立一对匿名的已连接的套接字,建立的两个套接字描述符会放在sv[0]和sv[1]中。既可以从sv[0]写入sv[1]读出,又可以从sv[1]读入sv[0]写出,如果没有写入就读出则会生阻塞。用途:用来创建全双工通道,不过只局限于父子进程之间。
下面我们通过一段代码来看看这个函数的用法:
#include <stdio.h> #include <errno.h> #include <string.h> #include <unistd.h> #include <sys/types.h> #include <sys/socket.h> int main() { int sv[2]={0,0}; int sock=socketpair(AF_LOCAL,SOCK_STREAM,0,sv); if(sock<0) { perror("socketpair"); exit(0); } pid_t id=fork(); char buf[1024]; if(id<0) { perror("fork"); exit(0); } else if(id==0) { close(sv[0]); //子进程关闭读端 while(1) { memset(buf,'\0',sizeof(buf)); strcpy(buf,"i am your child"); write(sv[1],buf,strlen(buf));//子进程写入 memset(buf,'\0',sizeof(buf)); ssize_t _s=read(sv[1],buf,sizeof(buf)-1);//子进程读取父进程的内容 buf[_s]='\0'; printf("father-->child:%s\n",buf); sleep(1); } close(sv[1]);//子进程关闭读端 } else {//父进程 close(sv[1]); while(1) { memset(buf,'\0',sizeof(buf)); ssize_t _s=read(sv[0],buf,sizeof(buf)-1); buf[_s]='\0'; printf("child-->father:%s\n",buf); memset(buf,'\0',sizeof(buf)); strcpy(buf,"i am your father"); write(sv[0],buf,strlen(buf)); sleep(1); } close(sv[0]); } return 0; }
我们看看运行结果:
我们可以看到,父子进程通过socketpair函数创建的全双工管道实现了进程间的通信,彼此都能收发信息,但要注意:父子进程在通信的时候,必须关闭一个描述符号,是因为一个在写的时候另一个只能读。