嵌入式学习-网络编程-Day5

这篇具有很好参考价值的文章主要介绍了嵌入式学习-网络编程-Day5。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

嵌入式学习-网络编程-Day5

一、思维导图

嵌入式学习-网络编程-Day5,学习,网络,php文章来源地址https://www.toymoban.com/news/detail-806985.html

二、作业

1.使用poll实现TCP服务器的并发

include <myhead.h>
//tcp服务器端
int main(int argc, const char *argv[])
{
	//1 创建通信套接字
	int sfd=-1;
	if((sfd=socket(AF_INET,SOCK_STREAM,0))==-1)
	{
		perror("socket error");
		return -1;
	}
	//快速重用端口号
	int reuse = 1;
	if(setsockopt(sfd,SOL_SOCKET,SO_REUSEADDR,&reuse,sizeof(reuse))==-1)
	{
		perror("setsockopt error");
		return -1;
	}

	//2 绑定IP地址和端口号
	struct sockaddr_in sin;  //地址信息结构体  man 7 ip
	sin.sin_family = AF_INET;//通信域 ipv4
	sin.sin_port = htons(8888);//端口号 转为网络字节序 2字节
	sin.sin_addr.s_addr = inet_addr("192.168.122.39");//ip地址
	if(bind(sfd,(struct sockaddr*)&sin,sizeof(sin))==-1)//绑定
	{
		perror("bind error");
		return -1;
	}

	//3 监听
	if(listen(sfd,128)==-1)
	{
		perror("listen error");
		return -1;
	}

	//4 接受客户端链接请求
	struct sockaddr_in cin;          //用于接收地址信息
	socklen_t socklen = sizeof(cin); //用于接收地址信息的大小
	
	//定义地址信息结构体数组,存储链接的客户端信息
	struct sockaddr_in cin_arr[1024];

	//定义一个等待文件描述符结构体数组
	int count_fd=4;  //文件描述符的个数
	struct pollfd pfd[count_fd];
	pfd[0].fd=0;
	pfd[3].fd=sfd;

	pfd[0].events=POLLIN;
	pfd[3].events=POLLIN;
	
	int maxfd=sfd;  //存储最大的文件描述符
	int newfd=-1;
	
	char rbuf[128]="";
	char wbuf[128]="";
	while(1)
	{
		int flag=poll(pfd,count_fd,-1);
		
		if(flag<0)
		{
			perror("poll error");
			return -1;
		}
		
		//1 新客户端链接 
		if(pfd[sfd].revents=POLLIN)
		{
			newfd=accept(sfd,(struct sockaddr*)&cin,&socklen);
			if(newfd==-1)
			{
				perror("accept error");
				return -1;
			}
			cin_arr[newfd]=cin;
			printf("[%s:%d]发来连接请求,newfd=%d\n",inet_ntoa(cin_arr[newfd].sin_addr),ntohs(cin_arr[newfd].sin_port),newfd);
			pfd[newfd].events=POLLIN;
			pfd[newfd].fd=newfd;
			//更新描述符的长度
			if(newfd>maxfd)
			{
				maxfd=newfd;
				count_fd++;
			}
		}

		//向客户端发送消息
		if(pfd[0].revents==POLLIN)
		{
			memset(wbuf,0,sizeof(wbuf));
			//scanf(" %s",wbuf);
			fgets(wbuf,sizeof(wbuf),stdin);
			wbuf[strlen(wbuf)-1]=0;
			
			for(int i=4;((i<=count_fd)&&(pfd[i].fd!=-1));i++)
			{
				//sendto(i,wbuf,sizeof(wbuf),0,(struct sockaddr*)&cin_arr[i],sizeof(cin_arr[i]));
				send(i,wbuf,sizeof(wbuf),0);
			}
			printf("发送成功\n");
		}

		//接收客户端的消息
		for(int i=4;i<=maxfd&&pfd[i].fd!=-1;i++)
		{
			if(pfd[i].revents==POLLIN)
			{
				//接收客户端消息
				memset(rbuf,0,sizeof(rbuf));
				//int res=read(i,rbuf,sizeof(rbuf));
				int res=recv(i,rbuf,sizeof(rbuf),0);
				if(res==0)
				{
					printf("客户端已经下线\n");
					close(i);
					//更新pfd 从集合中删除
					pfd[i].fd=-1;
				}
				printf("[%s:%d]:%s\n",inet_ntoa(cin_arr[i].sin_addr),\
						ntohs(cin_arr[i].sin_port),rbuf);
				//给客户端发消息
				strcat(rbuf,"^_^");
				send(i,rbuf,sizeof(rbuf),0);
				printf("发送成功\n");
			}
		}
	}
	return 0;
}

使用select实现TCP客户端的并发

#include <myhead.h>
//TCP 客户端
#define SER_IP "192.168.122.39"  //服务器ip地址
#define SER_PORT 8888          //服务器端口号

int main(int argc, const char *argv[])
{
	//1 创建套接字
	int cfd=-1;
	if((cfd=socket(AF_INET,SOCK_STREAM,0))==-1)
	{
		perror("socket error\n");
		return -1;
	}
	//2 可以不绑定
	//2.1填写地址信息结构体
	struct sockaddr_in cin;
	cin.sin_family=AF_INET;
	cin.sin_port=htons(6666);
	cin.sin_addr.s_addr=inet_addr("192.168.122.39");
	//2.2绑定
	if(bind(cfd,(struct sockaddr*)&cin,sizeof(cin))==-1)
	{
		perror("bind error");
		return -1;
	}
	//3 连接服务器
	struct sockaddr_in sin;
	sin.sin_family=AF_INET;
	sin.sin_port=htons(8888);
	sin.sin_addr.s_addr=inet_addr("192.168.122.39");

	if(connect(cfd,(struct sockaddr*)&sin,sizeof(sin))==-1)
	{
		perror("connect error");
		return -1;
	}
	printf("connect success\n");
	
	//准备文件描述符容器
	fd_set readfds,tempfds;
	FD_ZERO(&readfds);
	FD_SET(0,&readfds);
	FD_SET(cfd,&readfds);
	int maxfd=cfd;
	
	//4 收发数据
	char buf[128]="";
	while(1)
	{
		tempfds=readfds;
		int res=select(maxfd+1,&tempfds,NULL,NULL,NULL);
		if(res==-1)
		{
			perror("select error");
			return -1;
		}else if(res==0)
		{
			printf("timeout\n");
			return -1;
		}
		if(FD_ISSET(0,&tempfds))
		{
			//发送数据
			memset(buf,0,sizeof(buf));
			read(0,buf,sizeof(buf));
			buf[strlen(buf)-1]=0;
			if(strcmp(buf,"quit")==0)
			{
				printf("已退出\n");
				close(cfd);
				break;
			}
			write(cfd,buf,sizeof(buf));
			printf("发送成功\n");
		}
		if(FD_ISSET(cfd,&tempfds))
		{
			//接收数据
			recv(cfd,buf,sizeof(buf),0);
			printf("[%s:%d]:%s\n",SER_IP,SER_PORT,buf);
		}
	}
	return 0;
}

到了这里,关于嵌入式学习-网络编程-Day5的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包