C++局域网从服务器获取已连接用户的列表(linux to linux)

目录

服务器端

代码

客户端

代码解析

服务器端

原理

遇到的阻碍以及解决办法

客户端

原理

遇到的阻碍以及解决办法

运行结果截图

总结


服务器端

代码

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'为止,可能造成输出乱码。
  • 这里可以设置超时时间,因为它只是监听套接字和输入流,没有上面服务器端一样的遍历。

运行结果截图

总结

一个基于

C++入门教程(18)socket 实现简单聊天室_std socket_爱我呦呦的博客-CSDN博客文章浏览阅读8.6k次,点赞8次,收藏88次。本节通过socket实现一个简单的聊天室功能聊天室中如果有人说话,则服务器负责将内容传送给聊天室的其他人那么就需要客户端和服务端两个程序,客户端负责发送消息,服务端负责接收和转发消息客户端代码:#include #include #include #include #include #in_std sockethttps://blog.csdn.net/u011416077/article/details/123593428的拓展功能。并且把一些不足之处给指出来了。

相关推荐
Ajiang282473530432 分钟前
对于C++中stack和queue的认识以及priority_queue的模拟实现
开发语言·c++
中云DDoS CC防护蔡蔡34 分钟前
微信小程序被攻击怎么选择高防产品
服务器·网络安全·微信小程序·小程序·ddos
HPC_fac130520678161 小时前
以科学计算为切入点:剖析英伟达服务器过热难题
服务器·人工智能·深度学习·机器学习·计算机视觉·数据挖掘·gpu算力
速盾cdn3 小时前
速盾:CDN是否支持屏蔽IP?
网络·网络协议·tcp/ip
yaoxin5211233 小时前
第二十七章 TCP 客户端 服务器通信 - 连接管理
服务器·网络·tcp/ip
‘’林花谢了春红‘’5 小时前
C++ list (链表)容器
c++·链表·list
sinat_384241097 小时前
使用 npm 安装 Electron 作为开发依赖
服务器
机器视觉知识推荐、就业指导7 小时前
C++设计模式:建造者模式(Builder) 房屋建造案例
c++
Kkooe8 小时前
GitLab|数据迁移
运维·服务器·git