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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值