Linux进程间通信——Socket套接字
先来说下socket套接字技术,socket是一种接口技术,抽象成一个文件操作,可以让同一台计算机的进程之间通信,也可以让不同计算机的进程通信,或者说是网络通信,因此套接字的应用也是比较重要的。socket进程间通信底层需要借助socket文件,进行同一计算机下的进程间通信。至于是不是进程间通信或是网络通信,取决于socket函数中的参数,AF_UNIX / AF_LOCAL即本地通信、进程间通信,AF_INET是基于IPv4地址通信,AF_INET6则是基于IPv6地址通信。
好的介绍到这下面来看一下本地通信编程模型(数据流):
进程A | 进程B |
创建socket | 创建socket |
准备通信地址 (本地socket地址) | 准备对方通信地址 |
绑定socket和地址 | —— |
开启监听 | —— |
等待连接 | 连接 |
接收 / 发送数据 | 发送 / 接收数据 |
关闭socket | 关闭socket |
删除socket | —— |
其中,进程B是初识发送方,所以需要主动向进程A发出连接请求并先发送数据,而后进程A返回或者响应来自进程B的请求和数据。下面来看一下具体的实现:
下面来按照编程模型来实现进程A:
#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <unistd.h>
#include <sys/un.h>
#include <stdlib.h>
int main(int argc,const char* argv[])
{
//创建套接字
int sockfd=socket(AF_LOCAL,SOCK_STREAM,0);
if(sockfd<0)
{
perror("socket");
return EXIT_FAILURE;
}
//准备通信地址
struct sockaddr_un addr={};
addr.sun_family=AF_LOCAL;
strcpy(addr.sun_path,"sock");
socklen_t addrlen=sizeof(addr);
//绑定socket和通信地址
if(bind(sockfd,(struct sockaddr *)&addr,addrlen))
{
perror("bind");
return -1;
}
//监听
if(listen(sockfd,5))
{
perror("listen");
return -1;
}
//等待连接
int fd=accept(sockfd,(struct sockaddr *)&addr,&addrlen);
if(fd<0)
{
perror("accept");
return -1;
}
printf("有人连接成功!\n");
char buf[4096];
size_t buf_size=sizeof(buf);
while(1)
{
printf("recv……");
fflush(stdout);
//接收数据
size_t ret=read(fd,buf,buf_size);
if(0==strcmp("quit",buf))
{
printf("通信结束!\n");
break;
}
//发送数据
printf("\nrecv:%s\n>>>",buf);
scanf("%s",buf);
write(fd,buf,buf_size);
if(0==strcmp("quit",buf))
{
printf("通信结束!\n");
break;
}
}
//关闭socket
close(fd);
close(sockfd);
//删除socket
unlink(addr.sun_path);
return 0;
}
再按照编程模型来实现进程B:
#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <unistd.h>
#include <sys/un.h>
#include <stdlib.h>
int main(int argc,const char* argv[])
{
//创建套接字
int sockfd=socket(AF_LOCAL,SOCK_STREAM,0);
if(sockfd<0)
{
perror("socket");
return -1;
}
//准备对方通信地址
struct sockaddr_un addr={};
addr.sun_family=AF_LOCAL;
strcpy(addr.sun_path,"sock");
socklen_t addrlen=sizeof(addr);
//连接
if(connect(sockfd,(struct sockaddr *)&addr,addrlen))
{
perror("connect");
return -1;
}
//发送/接收
char buf[4096];
size_t buf_size=sizeof(buf);
while(1)
{
printf("\nrecv:%s\n>>>",buf);
scanf("%s",buf);
write(sockfd,buf,buf_size);
if(0==strcmp("quit",buf))
{
printf("通信结束\n");
break;
}
printf("recv……");
fflush(stdout);
size_t ret=read(sockfd,buf,buf_size);
if(0==strcmp("quit",buf))
{
printf("通信结束\n");
break;
}
}
//关闭socket
close(sockfd);
return 0;
}
测试结果如下:
over