【嵌入式学习】网络编程day0307---基于多路复用IO的聊天室

客户端

#include <myhead.h>
//创建结构体存储用户信息
struct User
{
	char tape;
	char username[32];
	char msg[1024];
};

int main(int argc, const char *argv[])
{
	//创建套接字与服务器进行通信
	int sfd=socket(AF_INET,SOCK_STREAM,0);
	if(sfd==-1)
	{
		perror("socket error");
		return -1;
	}

	//绑定
	//填充服务器结构体数据
	struct sockaddr_in sin;
	sin.sin_family=AF_INET;
	sin.sin_port=htons(8888);
	sin.sin_addr.s_addr=inet_addr("192.168.125.86");

	char username[32]="";
	struct User user;
	strcpy(&user.tape,"L");

	//创建容器存储文件标识符
	fd_set readfds,tempfds;
	//清空容器
	FD_ZERO(&readfds);
	FD_SET(0,&readfds);
	FD_SET(sfd,&readfds);
	int maxfd=sfd;
	while(1)
	{
		tempfds=readfds;
		int res=select(maxfd+1,&tempfds,NULL,NULL,NULL);
		if(res==-1)
		{
			perror("select error");
			return -1;
		}
		for(int i=0;i<=maxfd;i++)
		{
			if(!FD_ISSET(i,&tempfds))
			{
				continue;
			}	
			//向服务器读取消息
			else if(sfd==i)
			{
				if(user.tape=='L')
				{
					//向服务器发送数据(用户结构体)
					printf("请输入登录用户>>>");
					scanf("%s",username);
					strcpy(user.username,username);
					if(connect(sfd,(struct sockaddr*)&sin,sizeof(sin))==-1)
					{
						perror("connect error");
						return -1;
					};
					//发送只有用户名和结构类型
					send(sfd,&user,sizeof(user),0);
					user.tape='C';
				}
				else if(user.tape=='C')
				{
					char Msg[2046]="";
					int len=recv(sfd,Msg,sizeof(Msg),0);
					if(len>=0)
					{
						printf("%s",Msg);
					}

					else if(len==-1)
					{
						perror("recv error");
					}
				}
			}
			//向服务器发送数据
			else if(i==0&&user.tape=='C')
			{
				char msg[1024]="";
				//向服务器发送数据(用户结构体)
				scanf("%s",msg);	
				strcpy(user.msg,msg);
				if(strcmp(user.msg,"quit")==0)
				{
					user.tape='Q';
					send(sfd,&user,sizeof(user),0);
					goto A;
				}
				send(sfd,&user,sizeof(user),0);
			}

		}
	}
A:
	close(sfd);
	return 0;
}

服务器

#include <myhead.h>
struct userdata
{
	char tape;
	char username[32];
	char msg[1024];
};



int main(int argc, const char *argv[])
{

	//创建套接字文件与客户端连接
	int sfd=socket(AF_INET,SOCK_STREAM,0);
	if(sfd==-1)
	{
		perror("socket error");
		return -1;
	}
	signal(SIGPIPE, SIG_IGN);

	//绑定服务器信息
	//填充服务器信息
	struct sockaddr_in sin;
	sin.sin_family=AF_INET;
	sin.sin_port=htons(8888);
	sin.sin_addr.s_addr=inet_addr("192.168.125.86");
	socklen_t ssocklen=sizeof(sin); 

	//重新启用端口
	int resue=1;
	setsockopt(sfd,SOL_SOCKET,SO_REUSEADDR,&resue,sizeof(resue));
	//绑定
	if(bind(sfd,(struct sockaddr*)&sin,sizeof(sin))==-1)
	{
		perror("bind error");
		return -1;
	}
	char mbuf[2048]="";

	//监听
	if(listen(sfd,168)==-1)
	{
		perror("listen error");
		return -1;
	}
	//定义容器接收对端地址信息结构体
	struct sockaddr_in cin;	
	socklen_t socklen=sizeof(cin);
	int newfd=-1;
	char sbuf[128]="";


	//创建容器存储文件描述符
	fd_set readfds,tempfds;
	//清空容器
	FD_ZERO(&readfds);
	FD_SET(0,&readfds);
	FD_SET(sfd,&readfds);

	int maxfd=sfd;
	struct sockaddr_in cin_arr[1024];
	while(1)
	{
		//保留容器
		tempfds=readfds;
		int res=select(maxfd+1,&tempfds,NULL,NULL,NULL);
	//	printf("res=%d\n",res);
		if(res==-1)
		{
			perror("select error");
			return -1;
		}
		else if(res==0)
		{
			printf("TIME OUT\n");
			return -1;
		}

		for(int i=0;i<=maxfd;i++)
		{
			struct userdata Users;
			char Tape=Users.tape;	
			if(!FD_ISSET(i,&tempfds))
			{
				continue;
			}
			else if(i==sfd)  //表示sfd触发了事件
			{
				newfd=accept(sfd,(struct sockaddr*)&cin,&socklen);
				if(newfd==-1)
				{
					perror("accept error");
					return -1;
				}
				//服务器显示用户上线,并且广播给其他用户
  	
				recv(newfd,&Users,sizeof(struct userdata),MSG_DONTWAIT);
				memset(Users.msg,0,sizeof(Users.msg));
				Tape=Users.tape;	
				//更新maxfd
				if(newfd>maxfd)
				{	
					maxfd=newfd;
				}
				if(Tape=='L')
				{
					//将用户信息存储
					char str[128]="";
					sprintf(str,"用户[%s]已上线\n",Users.username);
					printf("%s",str);
					for(int j=4;j<=maxfd;j++)
					{
						if(j==newfd)
						{
							continue;
						}
						send(j,str,sizeof(str),0);
					}
				}
				//将客户端放入数组
				FD_SET(newfd,&readfds);
				tempfds=readfds;
				cin_arr[newfd]=cin;
				break;
			}
			else if(i==0)
			{
				fgets(sbuf,sizeof(sbuf),stdin);
				sbuf[strlen(sbuf)-1]=0;
				printf("触发键盘输入\n");
				char ssbuf[1024]="";
				sprintf(ssbuf,"服务器发来消息:%s\n",sbuf);
				for(int k=4;k<=maxfd;k++)
				{
					send(k,ssbuf,strlen(ssbuf),0);
				}
			}
			else
			{
				//与客户端进行数据收发
				//定义一个结构体存储客户端发来的正文
				recv(i,&Users,sizeof(struct userdata),MSG_DONTWAIT);
				Tape=Users.tape;
				printf("%c\n",Tape);
				//接收客户端发来的数据
				if(Tape=='C')
				{
					//创建数组
					char mbuf[2046]="";
					sprintf(mbuf,"用户[%s]发送消息:%s\n",Users.username,Users.msg);
					printf("%s\n",mbuf);
					for(int j=4;j<=maxfd;j++)
					{
						if(j==i)
						{
							continue;
						}
						send(j,mbuf,sizeof(mbuf),0);
					}
				}
				else if(Tape=='Q')
				{
					//创建数组
					char msg[1024]="";
					sprintf(msg,"用户[%s]已下线\n",Users.username);
					printf("%s\n",msg);
					for(int j=4;j<=maxfd;j++)
					{
						if(j==i)
						{
							continue;
						}
						send(j,msg,sizeof(msg),0);
					}
					//关闭套接字
					close(i);
					//将当前文件描述符从容器中移除
					FD_CLR(i,&readfds);
					//更新maxfd
					for(int j=maxfd;j>=0;j--)
					{
						if(FD_ISSET(j,&readfds))
						{
							maxfd=j;
							break;
						}
					}
					continue;
				} 
			}
		}
	}
	return 0;
} 

Ubuntu 18.04 PZB - 聊天室现象

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值