IO多路复用-超时检测-抓包分析-数据库

IO多路复用实现TCP服务器并发(未优化程序版)

#include<head.h>
int main(int argc, const char *argv[])
{
	//使用IO多路复用实现服务端
	//创建端点
	int sfd=socket(AF_INET,SOCK_STREAM,0);
	//参数1表示使用ipv4通信域
	//参数2表示使用TCP面向连接的通信方式
	//参数3表示补充通信协议,由于第二个参数已经指定通信方式,故写0
	if(sfd==-1){
		perror("socket");
		return 1;
	}
	else{
		printf("创建端点成功\n");
	}
	//端口快速重用
	int reluse=1;
	int rtsetsockopt=setsockopt(sfd,SOL_SOCKET,SO_REUSEADDR,&reluse,sizeof(reluse));
	if(rtsetsockopt==-1){
		perror("setsockopt");
		return -1;
	}

	//将套结文件与IP地址与端口绑定
	struct sockaddr_in ip_port;
	char* ip="192.168.176.130";
	uint16_t port=9999;
	ip_port.sin_family=AF_INET;
	ip_port.sin_port=htons(port);
	ip_port.sin_addr.s_addr=inet_addr(ip);
	int retbind=bind(sfd,(struct sockaddr*)&ip_port,sizeof(ip_port));
	if(retbind==-1){
		perror("bind");
		return 1;
	}
	else if(retbind==0){
		printf("端口和IP绑定成功\n");
	}
	//将套接字设置为被动监听模式
	int retlisten=listen(sfd,128);
	if(retlisten==-1){
		perror("listen");
		return 1;
	}
	else if(retlisten==0){
		printf("设置监听成功\n");
	}
	//阻塞等待客户端连接
	//1.准备检测文件描述符的集合
	fd_set reader;
	fd_set temp;
	//清空集合中的内容
	FD_ZERO(&reader);
	//将检测的文件描述符放入文件中
	FD_SET(0,&reader);
	FD_SET(sfd,&reader);
	int maxsfd=sfd;
	struct sockaddr_in ser_ip_port;
	socklen_t addrlen=sizeof(ser_ip_port);
	struct sockaddr_in cin_addr[1024];//ser_ip_port变量每次客户连接都会更新一次,只能保留最后一个客户端信息,故建立用于存放客户端地址结构体信息数组
	bzero(cin_addr,sizeof(cin_addr));
	int retaccept=0;
	while(1){
		//检测前将readerer备份,防止select函数检测到时间发生后,在reader中只保留发生的文件描述符,而下次循环时文件描述符丢失
		temp=reader;
	//	printf("maxsfd=%d\n",maxsfd);
		//使用select函数检测文件描述集合中是否有事件产生,第一个参数是文件描述符最大值+1
		int retselect=select(maxsfd+1,&temp,NULL,NULL,NULL);
		if(retselect==-1){
			perror("select");
			return -1;
		}
		else if(retselect==0){
			printf("time out\n");
		}
		//程序执行到此,说明已经有事件发生,select函数已经解除阻塞
		//判断sfd文件符事件
		if(FD_ISSET(sfd,&temp)==1){
			retaccept=accept(sfd,(struct sockaddr*)&ser_ip_port,&addrlen);
			if(retaccept==-1){
				perror("accept");
				return 1;
			}
			else{
				printf("客户端连接成功\n");
				printf("客户端IP:%s 客户端端口:%d\n",inet_ntoa(ser_ip_port.sin_addr),ntohs(ser_ip_port.sin_port));
			}
			//将retaccept文件描述符放入到reader中,用于检测是否有客户端发来数据
			FD_SET(retaccept,&reader);
		//	printf("%d\n",retaccept);
			//有新的文件描述符加入,需更新select函数第一个参数值
			if(retaccept>=maxsfd){
				maxsfd=retaccept;
			}
			cin_addr[retaccept]=ser_ip_port;
		}
		//与客户端相互通信
		for(int i=4;i<=maxsfd;i++){//执行到此说明有retaccept事件发生,遍历找到发生的事件,如果没有遍历,
			//先连接的retaccept不会被找到(之间连接的,产生retaccept事件,不会经过accept重新产生retaccept),
			//retaccept的值一直都是最后连接accept产生的
		if(FD_ISSET(i,&temp)){//遍历找到发生的retaccpt事件
			//printf("%d\n",retaccept);
			char rbuf[128]={0};
			bzero(rbuf,128);
			int retread=recv(i,rbuf,128,0);
			if(retread==0){//套结断开连接
				printf("客户端断开连接\n");
				close(i);//关闭与客户端建立连接的套接字
				//套接不再使用,不从容器中清除,会影响select第一个参数,需将其从文件清除
				FD_CLR(i,&reader);
				//无需删除cin_addr[retaccept],文件描述符是最小未使用分配原则,如果退出,下次有新的进来会覆盖原来的内容
				//更新maxsfd
				for(int j=maxsfd;j>=sfd;j--){
					if(FD_ISSET(j,&reader)==1){//如果删除的恰好是maxsfd,则在容中从大到小器遍历找到下一个最大描述符,没删除则保持不变
						maxsfd=j;
						break;
					}
				}
				continue;//结束本次循环,进入下次
			}
			printf("[%s:%d]读取客户端内容:%s\n",inet_ntoa(cin_addr[i].sin_addr),ntohs(cin_addr[i].sin_port),rbuf);
		}
		}
		if(FD_ISSET(0,&temp)){
			for(int j=4;j<=maxsfd;j++){
				if(FD_ISSET(j,&reader))
				{
					char wbuf[128]={0};
					bzero(wbuf,128);
					scanf("%s",wbuf);
					getchar();
					write(j,wbuf,128); 
					printf("成功回复客服端\n");
				}
			}
		}
	}
	close(sfd);

	return 0;
} 

IO多路复用实现TCP客户端收发并发

#include<head.h>
int main(int argc, const char *argv[])
{
	//创建一个客户端,使用poll函数实现并发
	//创建端点
	int sfd=socket(AF_INET,SOCK_STREAM,0);
	if(sfd==-1){
		perror("socket");
		return 1;
	}
	else{
		printf("创建成功\n");
	}
	//端口快速重用
	int reluse=1;
	int rtsetsockopt=setsockopt(sfd,SOL_SOCKET,SO_REUSEADDR,&reluse,sizeof(reluse));
	if(rtsetsockopt==-1){
		perror("setsockopt");
		return -1;
	}
	//将套结文件与IP,端口绑定
	struct sockaddr_in ip_port;
	char* ip="192.168.176.130";
	uint16_t port=8888;
	ip_port.sin_family=AF_INET;
	ip_port.sin_port=htons(port);
	ip_port.sin_addr.s_addr=inet_addr(ip);
	int retbind=bind(sfd,(struct sockaddr*)&ip_port,sizeof(ip_port));
	if(retbind==-1){
		perror("bind");
		return 1;
	}
	else if(retbind==0){
		printf("IP与端口绑定成功\n");
	}
	//与服务器连接
	struct sockaddr_in ip_port_ser;
	char* ip_ser="192.168.176.130";
	uint16_t port_ser=9999;
	ip_port_ser.sin_family=AF_INET;
	ip_port_ser.sin_port=htons(port_ser);
	ip_port_ser.sin_addr.s_addr=inet_addr(ip_ser);
	int retconnect=connect(sfd,(struct sockaddr*)&ip_port_ser,sizeof(ip_port_ser));
	if(retconnect==-1){
		perror("connect");
		return 1;
	}
	else if(retconnect==0){
		printf("连接成功\n");
	}
	//使用poll函数实现客户端并发
	struct pollfd fds[2];
	fds[0].fd=0;//检测输入事件
	fds[0].events=POLLIN;//检测读事件
	fds[1].fd=sfd;//检测sfd文件描述符,读取服务器消息
	fds[1].events=POLLIN;//检测读事件
	//与服务器相互通信
	char rbuf[128]={0};
	char wbuf[128]={0};
	printf("请向服务端发送信息:\n");
	while(1){
		int rtpoll=poll(fds,2,-1);//最后一个参数表示延时,小于0表示阻塞,监测事件
		if(rtpoll==-1){
			perror("poll");
			return -1;
		}
		memset(rbuf,0,128);
		memset(wbuf,0,128); 
		if(fds[0].revents==POLLIN){//检测到客户端输入事件
			fgets(wbuf,128,stdin);
			wbuf[strlen(wbuf)-1]=0;
			send(sfd,wbuf,sizeof(wbuf),0);
			printf("向服务端发送消息成功\n");
		}
		if(fds[1].revents==POLLIN){//检测到服务器发送事件
			ssize_t retrecv=recv(sfd,rbuf,128,0);//如果套接断开连接会变成非阻塞
			if(retrecv==0){break;}
			printf("接受服务端信息:%s\n",rbuf);
		}
	}
	close(sfd);
	return 0;
}

select与poll函数超时设置

#include<head.h>
int main(int argc, const char *argv[])
{
	/***************使用select第五个参数实现超时*/
	//定义文件描述符集合
	/*fd_set readers;
	//清空集合
	FD_ZERO(&readers);
	//将检测的文件描述符放入集合;
	FD_SET(0,&readers);
	//设置延时时间
	struct timeval tv={5,0};//秒,微秒
	while(1){
		tv.tv_sec=5;
		tv.tv_usec=0;//tv为地址传递,内部自动计时,每次循环后的值会改变,所以手动初始
		int rtselesct=select(1,&readers,NULL,NULL,&tv);
		if(rtselesct==-1){
			perror("select");
			return -1;
		}else if(rtselesct==0){
			printf("timeout\n");
			return -1;
		}
		if(FD_ISSET(0,&readers)){
			int num=0;
			scanf("%d",&num);
			printf("%d\n",num);
		}
	}*/
	/****使用poll第三个参数实现超时********/
	//定义文件描述符集合
	struct pollfd pfd;
	pfd.fd=0;//检测0号文件描述符
	pfd.events=POLLIN;//检测读事件
	while(1){
		int rtpoll=poll(&pfd,1,5000);//微秒,第三个参数是值传递
		if(rtpoll==-1){
			perror("poll");
			return -1;
		}else if(rtpoll==0){
			printf("timeout\n");
			return -1;
		}
		if(pfd.revents==POLLIN){
			int num=0;
			scanf("%d",&num);
			printf("%d\n",num);
		}
	}
		
	return 0; 
} 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值