服务器端和客户端在执行时 都需在后边加上IP地址和端口号
(客户端后面加的是服务器的IP地址和端口号)
服务器端代码
#include
int main(int argc, const char *argv[])
{
if(argc < 3)
{
printf("请输入 ip 和端口号\n");
return -1;
}
//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;
sin.sin_family = AF_INET;
sin.sin_port = htons(atoi(argv[2]));
sin.sin_addr.s_addr = inet_addr(argv[1]);
//2.2绑定工作
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、接收链接请求
//4.1 定义用于接收客户端地址信息的结构体变量
struct sockaddr_in cin;
socklen_t socklen = sizeof(cin); //接收长度
//4.2 接收客户端链接请求
int newfd = -1; //用于跟客户端通信的套接字文件描述符
char buf[128] = ""; //用于终端输入的字符串
char wbuf[128]="**system**:";
//定义检测文件描述符的集合
fd_set readfds;
//清空集合
FD_ZERO(&readfds);
//将sfd和0文件描述符放入检测集合中
FD_SET(sfd, &readfds);
FD_SET(0, &readfds);
int res = 0; //接收select的返回值
fd_set tempfds; //定义一个临时集合
int maxfd = sfd; //存放集合中的最大文件描述符
char rbuf[128] = ""; //读取消息的容器
struct sockaddr_in arr_cin[1024]; //用于存储客户端的地址信息结构体
while(1)
{
tempfds = readfds;
//使用select检测集合中释放有事件产生
res = select(maxfd+1, &tempfds, NULL, NULL, NULL);
if(res == -1)
{
perror("select error");
return -1;
}else if(res == 0)
{
printf("time out\n");
return -1;
}
//当程序执行到该处,说明,集合中已经有文件描述符产生事件
for(int index=0; index<=maxfd; index++)
{
//判断当前的文件描述符是否不在集合中,如果不在直接跳过
if(!FD_ISSET(index, &tempfds))
{
continue;
}
//判断是否因客户端连接触发了事件
if( index == sfd )
{
//接收客户端链接请求
if( (newfd=accept(sfd, (struct sockaddr*)&cin, &socklen)) == -1)
{
perror("accept error");
return -1;
}
//将当前的客户端存放到下标为newfd的位置
arr_cin[newfd] = cin;
bzero(rbuf,sizeof(rbuf));
printf("[%s:%d]登录成功\n", inet_ntoa(arr_cin[newfd].sin_addr),\
ntohs(arr_cin[newfd].sin_port));
//将newfd放入待检测集合中
FD_SET(newfd, &readfds);
//更新 maxfd
if(newfd > maxfd)
{
maxfd = newfd;
}
}else if( index == 0 )//判断0号文件描述符是否触发了事件
{
scanf("%s", buf);
strcat(wbuf,buf);
//printf("触发终端输入:%s\n", buf);
//将数据发送给所有客户端
for(int j=4; j<=maxfd&&FD_ISSET(j,&readfds); j++)
{
send(j, wbuf, sizeof(wbuf), 0);
}
bzero(wbuf,sizeof(wbuf));
strcat(wbuf,"**system**:");
}else
{
//判断是否为客户端发来消息,newfd触发的事件
//由于有多个客户端要处理,所以需要进行遍历
//5、收发数据
//清空内容
bzero(rbuf, sizeof(rbuf));
int res = recv(index, rbuf, sizeof(rbuf), 0);
if(res == 0)
{
close(index);
//移除newfd
FD_CLR(index, &readfds);
//更新maxfd
for(int i=maxfd; i>0; i--)
{
if(FD_ISSET(i, &readfds))
{
maxfd = i;
break;
}
}
continue;
}
printf("[%s:%d] : %s\n", inet_ntoa(arr_cin[index].sin_addr), ntohs(arr_cin[index].sin_port), rbuf);
//链接一个字符串后发回去
//将数据发送给所有客户端
for(int j=4; j<=maxfd&&FD_ISSET(j,&readfds); j++)
{
if(index==j)
{continue;}
send(j, rbuf, sizeof(rbuf), 0);
}
}
}
}
//关闭套接字
close(sfd);
return 0;
}
客户端代码
#include
int main(int argc, const char *argv[])
{
if(argc < 3)
{
printf("请输入 ip 和端口号\n");
return -1;
}
//1、创建套接字
int cfd = -1;
if((cfd = socket(AF_INET, SOCK_STREAM, 0)) == -1)
{
perror("socket error");
return -1;
}
//设置端口号快速重用
int reuse = 1;
if(setsockopt(cfd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse))==-1)
{
perror("setsockopt error");
return -1;
}
//2.绑定服务器ip和端口号
struct sockaddr_in cin;
cin.sin_family = AF_INET;
cin.sin_port = htons(atoi(argv[2]));
cin.sin_addr.s_addr = inet_addr(argv[1]);
if((connect(cfd,(struct sockaddr*)&cin,sizeof(cin)))==-1)
{
perror("connect error");
return -1;
}
printf("connect success\n");
struct pollfd fds[2];
fds[0].fd=0;
fds[0].events=POLLIN;
fds[1].fd=cfd;
fds[1].events=POLLIN;
char rbuf[128]="";
char buf[128]="";
printf("请输入姓名:");
scanf("%s",buf);
getchar();
sprintf(rbuf,"--------%s上线--------",buf);
send(cfd,rbuf,sizeof(rbuf),0);
char wbuf[128]="";
int res=-1;
while(1)
{
res=poll(fds,2,-1);
if(res<0)
{
perror("poll error");
return -1;
}
else if(res==0)
{
printf("time out\n");
return -1;
}
//发信息
if(fds[0].revents==POLLIN)
{
bzero(rbuf,sizeof(rbuf));
fgets(rbuf,sizeof(rbuf),stdin);
rbuf[strlen(rbuf)-1]=0;
if(strcmp(rbuf,"quit")==0)
{
bzero(wbuf,sizeof(wbuf));
sprintf(wbuf,"%s 已下线",buf);
send(cfd,wbuf,sizeof(wbuf),0);
goto END;
}
bzero(wbuf,sizeof(wbuf));
sprintf(wbuf,"%s:%s",buf,rbuf);
send(cfd,wbuf,sizeof(wbuf),0);
}
if(fds[1].revents==POLLIN)
{
bzero(wbuf,sizeof(wbuf));
recv(cfd,wbuf,sizeof(wbuf),0);
printf("%s\n",wbuf);
}
}
END:
close(cfd);
return 0;
}