Socket编程---p2p模型

本文介绍了如何使用Socket编程实现点对点(P2P)通信模型,讲解了在这种模型下,一个节点既可以作为服务端发送数据,也可以作为客户端接收数据。详细给出了服务端和客户端的代码示例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

自己既能发送数据也能接收数据,既是服务端也是客户端

在这里插入图片描述
服务端代码:

#include<unistd.h>
#include<sys/stat.h>
#include<sys/wait.h>
#include<sys/types.h>
#include<fcntl.h>

#include<stdlib.h>
#include<stdio.h>
#include<errno.h>
#include<string.h>
#include<signal.h>
#include<arpa/inet.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<sys/socket.h>


/*
typedef void (*sighandler_t)(int);

sighandler_t signal(int signum, sighandler_t handler);
*/
//信号要出发的回调函数
void handle(int num){
	printf("recv num:%d,退出\n",num);
	exit(0);
}

int main(){
	int sockfd=0;
	signal(SIGUSR1,handle);//捕获SIGUSR1信号,并调用handle函数
	sockfd=socket(AF_INET,SOCK_STREAM,0);
	
	if(sockfd==-1){
		perror("fun socket\n");
		exit(0);
	}
	struct sockaddr_in serveraddr;
	serveraddr.sin_family=AF_INET;
	serveraddr.sin_port=htons(8001);
	serveraddr.sin_addr.s_addr=inet_addr("127.0.0.1");
	//serveraddr.sin_addr.s_addr=INADDR_ANY;//绑定本机的任意地址	
	int optval=1;
	
    if(setsockopt(sockfd, SOL_SOCKET,SO_REUSEADDR,&optval, sizeof(optval))<0){
		perror("setsockopt");
		exit(0);
	};

	if(bind(sockfd, (const struct sockaddr *)&serveraddr,sizeof(serveraddr))<0){
		perror("bind()");
		exit(0);
	};
	//一旦调用了listen这个函数,那么这个套接字sockfd将变成被动套接字;只能接受连接,不能主动发送连接
	//listen做了两个队列1、已经由客户端发出并达到服务器,服务器正在等待完成相应的tcp三次握手过程
	//2、已完成连接的队列
	if(listen(sockfd, SOMAXCONN)<0){
		perror("listen()");
		exit(0);
	};
	//
	//struct sockaddr perraddr;//通用ip
	//socklen_t perrlen;
	struct sockaddr_in perraddr;//tcpip 地址结构
	int perrlen=sizeof(perraddr);
	int conn=0;		
	//accept 接受一个新的连接,这个新的连接是一个主动套接字
	conn=accept(sockfd, (struct sockaddr *)&perraddr,&perrlen);	
	if(conn==-1){
		perror("fun accept");
		exit(0);
	}
	printf("perradd:%s\n perrport:%d\n",inet_ntoa(perraddr.sin_addr),ntohs(perraddr.sin_port));
	int pid=fork();
	if(pid>0){
		//父进程读取客户端的数据
		char revbuf[100]={0};
		while(1){
			int ret=read(conn,revbuf,sizeof(revbuf));
			if(ret==0){
				//如果在读的过程中,对方已经关闭,已返回tcpip协议,会返回一个o数据报
				printf("对方已关闭\n");
				//如果对方已经关闭,break,结束循环。
				break;
			}else if(ret<0){
				printf("读数据失败");
				exit(0);
			}
			fputs(revbuf,stdout);//服务器端收到数据,打印屏幕
			memset(revbuf,0,sizeof(revbuf));
		}	
		close(conn);//关闭连接
		kill(pid,SIGUSR1);//并向子进程发送信号,结束子进程
		sleep(1);
	}else if(pid==0){
		//char revbuf[1024]={0};
		char sendbuf[1024]={0};
		while(fgets(sendbuf,sizeof(sendbuf),stdin)!=NULL){
			//向客户端写数据
			write(conn,sendbuf,strlen(sendbuf));//服务端回发信息
			memset(sendbuf,0,sizeof(sendbuf));
		}
	}
	close(sockfd);	
	return 0;
}

客户端代码:

#include<unistd.h>
#include<sys/stat.h>
#include<sys/wait.h>
#include<sys/types.h>
#include<fcntl.h>

#include<stdlib.h>
#include<stdio.h>
#include<errno.h>
#include<string.h>
#include<signal.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <sys/socket.h>

void handle(int num){
	printf("recv num:%d,退出\n",num);
	exit(0);
}

int main(){
	int sockfd=0;
	signal(SIGUSR1,handle);
	sockfd=socket(AF_INET,SOCK_STREAM,0);
	
	if(sockfd==-1){
		perror("fun socket\n");
		exit(0);
	}
	struct sockaddr_in serveraddr;
	serveraddr.sin_family=AF_INET;
	serveraddr.sin_port=htons(8001);
	serveraddr.sin_addr.s_addr=inet_addr("127.0.0.1");
	//serveraddr.sin_addr.s_addr=INADDR_ANY;//绑定本机的任意地址	
	
	//int connect(int sockfd, const struct sockaddr *addr,socklen_t addrlen);

	if(connect(sockfd, (const struct sockaddr *)&serveraddr,sizeof(serveraddr))<0){
		perror("fun connetc");
		exit(0);
	};
	int pid=fork();
	if(pid>0){//父进程从键盘接收数据
		//char revbuf[1024]={0};
		char sendbuf[1024]={0};
		
		while(fgets(sendbuf,sizeof(sendbuf),stdin)!=NULL){
			write(sockfd,sendbuf,strlen(sendbuf));//发送给服务端的父进程。
			memset(sendbuf,0,sizeof(sendbuf));
		}
	}else if(pid==0){
		//子进程接收数据
		while(1){
			int ret=0;
			char revbuf[1024]={0};
			//从服务器读数据
			ret=read(sockfd,revbuf,sizeof(revbuf));
			if(ret<0){
				printf("read err\n");
				break;
			}
			if(ret==0){
				printf("read err\n");
				break;
			}
			fputs(revbuf,stdout);//从服务器收到数据,打印屏幕
			memset(revbuf,0,sizeof(revbuf));
		}
		close(sockfd);
		kill(getppid(),SIGUSR1);//非父进程发信号,退出进程。
		exit(0);
	}else{
		printf("进程创建失败\n");
	}
	close(sockfd);	
	return 0;
}

客户端先退出:
在这里插入图片描述
服务端先退出:
在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值