目录
服务器端
代码
cpp
#include <sys/types.h>
#include <sys/socket.h>
#include <stdio.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>
#include <fcntl.h>
#include <sys/shm.h>
#include <iostream>
#include <thread>
#include <list>
using namespace std;
#define PORT 8806
#define IP "127.0.0.1"
int msocket;
struct sockaddr_in servaddr;
socklen_t m_len;
list<int> connList;
void addPerson(){
while(1){
int f_socket=accept(msocket,(struct sockaddr *) &servaddr,&m_len);
connList.push_back(f_socket);
cout<<"玩家"<<f_socket<<":进入房间"<<endl;
}
}
void sendList(int f_socket){
for(int i=0;i<=connList.size()-1;i++){
char buf_1[1024];
memset(buf_1,0,sizeof(buf_1));
sprintf(buf_1,"玩家%d(在线)\n",i+3);
send(f_socket,buf_1,sizeof(buf_1),0);
}
}
void recv_meg(){
struct timeval tv;
tv.tv_sec=0;
tv.tv_usec=0;
list<int>::iterator it;
while(true){
for(it=connList.begin();it!=connList.end();it++){
fd_set rds;
FD_ZERO(&rds);
int max_fd=0;
int reval=0;
FD_SET(*it,&rds);
if(*it>max_fd){
max_fd=*it;
}
reval=select(max_fd+1,&rds,NULL,NULL,&tv);
if(reval==-1){
cout<<"error"<<endl;
}
else if(reval==0){
//pass
}
else{
char buf[1024];
memset(buf,0,sizeof(buf));
int len_1=recv(*it,buf,sizeof(buf),0);
if(len_1>0){
if(buf[0]=='#'){
sendList(*it);
}
cout<<"收到"<<*it<<"玩家的消息:";
cout<<buf<<endl;
}
}
}
}
}
int main(){
msocket=socket(AF_INET,SOCK_STREAM,0);
memset(&servaddr,0,sizeof(servaddr));
servaddr.sin_family=AF_INET;
servaddr.sin_addr.s_addr=htonl(INADDR_ANY);
servaddr.sin_port=htons(PORT);
if(bind(msocket,(struct sockaddr*) &servaddr,sizeof(servaddr))==-1){
perror("bind");
exit(1);
}
if(listen(msocket,20)==-1){
perror("listen");
exit(1);
}
m_len=sizeof(servaddr);
thread thr_1(recv_meg);
thr_1.detach();
thread th_2(addPerson);
th_2.detach();
while(1){}
return 0;
}
客户端
cpp
#include <iostream>
#include <sys/socket.h>
#include <stdio.h>
#include <sys/time.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <cstring>
using namespace std;
#define PORT 8806
int main(){
int socket_1;
fd_set rds;
FD_ZERO(&rds);
struct timeval tv;
socket_1=socket(AF_INET,SOCK_STREAM,0);
struct sockaddr_in servaddr;
servaddr.sin_family=AF_INET;
servaddr.sin_port=htons(PORT);
servaddr.sin_addr.s_addr=inet_addr("127.0.0.1");
if(connect(socket_1,(struct sockaddr*)&servaddr,sizeof(servaddr))<0){
perror("connect");
exit(1);
}
while(1){
tv.tv_sec=10;
tv.tv_usec=0;
FD_ZERO(&rds);
FD_SET(0,&rds);
FD_SET(socket_1,&rds);
int max_fd=0;
int reval=0;
if(max_fd<socket_1){
max_fd=socket_1;
}
reval=select(max_fd+1,&rds,NULL,NULL,&tv);
if(reval==-1){
cout<<"error!"<<endl;
}
else if(reval==0){
//pass
}
else{
if(FD_ISSET(socket_1,&rds)){
char buf_2 [1024];
int len_1=recv(socket_1,buf_2,sizeof(buf_2,0),0);
if(len_1>0){
cout.write(buf_2,len_1);
}
memset(buf_2,0,sizeof(buf_2));
}
if(FD_ISSET(0,&rds)){
char buf_3 [1024];
fgets(buf_3,sizeof(buf_3),stdin);
send(socket_1,buf_3,sizeof(buf_3),0);
memset(buf_3,0,sizeof(buf_3));
}
}
}
return 0;
}
代码解析
服务器端
原理
创建了一个套接字绑定端口和ip地址
之后创建一个线程用于接收外面的连接请求并且把生成的客户端的套接字存储到connList的链表中
之后用在又创建以线程用select函数来判断是否有客户端进行发送请求,如果有判断第一个首字母是否为'#'如果是那么就放回发送connList链表
遇到的阻碍以及解决办法
- 发送的数据的时候最好加一个\n,因为客户端没有换行,所以打印的时候就会怪怪的
- 由于是第一次实战,开始那个大小端互相转化把我搞懵了,索性本来也就不难干脆就记下来就行了。
- 创建的服务器套接字以及客户端生成的套接字容易搞混,用命名把它分开来了。
- 超时时间目前服务器端来看没什么用反而会影响新客户端的连接,服务器端会遍历套接字列表,如果在等待的时候有新的连接进行请求依旧会继续堵塞(因为他一次只能select一个套接字)
客户端
原理
客户端就只有一个主函数,同样绑定套接字之后,用select用来判断是否有输入或者接受请求,如果有输入则发送输入字节流,如果有接收请求则调用接收函数并且将其打印出来
遇到的阻碍以及解决办法
- 打印输出的列表用printf函数或者用cout.write(字符串,长度)不要用cout函数会乱码------printf函数根据格式化字符串来输出数据,而cout在默认情况下将字符串视为null-terminated字符串,即以'\0'结尾。所以,如果buf_2中的数据不是以'\0'结尾的,cout将会继续输出直到遇到'\0'为止,可能造成输出乱码。
- 这里可以设置超时时间,因为它只是监听套接字和输入流,没有上面服务器端一样的遍历。
运行结果截图
总结
一个基于