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

目录

服务器端

代码

客户端

代码解析

服务器端

原理

遇到的阻碍以及解决办法

客户端

原理

遇到的阻碍以及解决办法

运行结果截图

总结


服务器端

代码

#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;
}

客户端

#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的拓展功能。并且把一些不足之处给指出来了。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/189866.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

安捷伦E4404B频谱分析仪,100 Hz 至 6.7 GHz

E4404B是安捷伦ESA-E系列频谱分析仪&#xff0c;它是一款能够适应未来发展需求的中高端频谱分析仪解决方案。该系列在频谱分析仪的测量速度、动态范围、精度和功率分辨能力等方面&#xff0c;都为类似价位的产品树立了性能标杆。其灵活的平台设计使得研发、制造和现场服务工程师…

这一款 Mac 系统终端工具,已经用的爱不释手了!

&#x1f525;&#x1f525;&#x1f525;作为程序员或者运维管理人员&#xff0c;我们经常需要使用终端工具来进行服务器管理及各种操作&#xff0c;比如部署项目、调试代码、查看/优化服务、管理服务器等。 相信大家用的最多的终端工具就是 Xshell、iTerm2和Mobaxterm&#…

利用ngrok实现内网穿透(全网最详细教程)

准备工具&#xff1a; 1、phpstudy 用于在本地搭建网站 2、ngrok 用于将自己的本地端口暴露到公网上&#xff0c;从而实现内网穿透 文章开始前给大家分享一个学习人工智能的网站&#xff0c;通俗易懂&#xff0c;风趣幽默 人工智能https://www.captainbed.cn/myon/ ~~~~~…

ESP32和ESP8266的ESP-MESH

ESP32和ESP8266的ESP-MESH 功能介绍一、介绍ESP-MESH二、安装painlessMesh库三、ESP-MESH基本示例&#xff08;广播消息&#xff09;四、示范 功能介绍 了解如何使用ESP-MESH网络协议通过ESP32和ESP8266 NodeMCU板构建网状网络。 ESP-MESH允许多个设备&#xff08;节点&#x…

单例模式与多线程

目录 前言 正文 1.立即加载/饿汉模式 2.延迟加载/懒汉模式 1.延迟加载/懒汉模式解析 2.延迟加载/懒汉模式的缺点 3.延迟加载/懒汉模式的解决方案 &#xff08;1&#xff09;声明 synchronized 关键字 &#xff08;2&#xff09;尝试同步代码块 &#xff08;3&am…

vue 中 js 金额数字转中文

参考&#xff1a;js工具函数之数字转为中文数字和大写金额_js封装工具类函数金额大写-CSDN博客 我使用的框架vol.core。 客户需求要将录入框的金额数字转换成中文在旁边显示&#xff0c;换了几种函数&#xff0c;最终确定如下函数 function changeToChineseMoney(Num) {//判断…

Quartz定时任务基础

springBoot有一个定时执行某个方法的 注解&#xff1a; Scheduled 可以满足挺多的需求&#xff0c;但是到了一些场景&#xff0c;就显得比较麻烦&#xff0c;比如&#xff1a; 机器待机五分钟后执行切换待机状态。如果是按照使用Scheduled注解&#xff0c;就得持久化一个表&…

【Java SE】 带你走近Java的抽象类与接口

&#x1f339;&#x1f339;&#x1f339;【JavaSE】专栏&#x1f339;&#x1f339;&#x1f339; &#x1f339;&#x1f339;&#x1f339;个人主页&#x1f339;&#x1f339;&#x1f339; &#x1f339;&#x1f339;&#x1f339;上一篇文章&#x1f339;&#x1f339;&…

2018年3月26日 Go生态洞察:Go包版本管理提案分析

&#x1f337;&#x1f341; 博主猫头虎&#xff08;&#x1f405;&#x1f43e;&#xff09;带您 Go to New World✨&#x1f341; &#x1f984; 博客首页——&#x1f405;&#x1f43e;猫头虎的博客&#x1f390; &#x1f433; 《面试题大全专栏》 &#x1f995; 文章图文…

java springboot测试类虚拟MVC环境 匹配请求头指定key与预期值是否相同

上文 java springboot测试类虚拟MVC环境 匹配返回值与预期内容是否相同 (JSON数据格式) 版 中 我们展示 json匹配内容的方式 那么 本文我们来看看Content-Type属性的匹配方式 首先 我们从返回体可以看出 Content-Type 在请求头信息 Headers 中 我们直接将测试类代码更改如下 …

C#,《小白学程序》第二十七课:大数四则运算之“运算符重载”的算法及源程序

1 文本格式 using System; using System.Text; using System.Collections; using System.Collections.Generic; /// <summary> /// 大数的四则&#xff08;加减乘除&#xff09;运算 /// 及其运算符重载&#xff08;取余数&#xff09; /// </summary> public cl…

在项目中集成marsUI

拷贝文件夹到目标项目 集成 安装相关依赖 npm i --save ant-design-vue4.x npm i less npm i nprogress npm i consola npm i echarts npm i vue-color-kit npm i icon-park/svg npm i vite-plugin-style-import 配置Vite文件 使用 效果

Leetcode—828.统计子串中的唯一字符【困难】

2023每日刷题&#xff08;四十一&#xff09; Leetcode—828.统计子串中的唯一字符 算法思想 枚举所有种类字母在s中出现的位置&#xff0c;分别统计只包含这个字母不包含该类字母中其他字母的子串个数 实现代码 int uniqueLetterString(char* s) {int len strlen(s);cha…

电子学会C/C++编程等级考试2022年06月(二级)真题解析

C/C++等级考试(1~8级)全部真题・点这里 第1题:小白鼠再排队 N只小白鼠(1 < N < 100),每只鼠头上戴着一顶有颜色的帽子。现在称出每只白鼠的重量,要求按照白鼠重量从小到大的顺序输出它们头上帽子的颜色。帽子的颜色用 “red”,“blue”等字符串来表示。不同的小白…

十分钟让你搞懂JVM中的GC垃圾回收机制(分代回收)

文章目录 0. 为什么要有垃圾回收?1. 垃圾回收哪个内存区域?2. 如何找到垃圾(死亡对象的判断)2.1 引用计数法2.2 可达性分析法2.3 两种算法的差别 3. 如何清理垃圾(死亡对象的回收)3.1 标记-清楚法3.2 复制法3.3 标记-整理法 4. JVM使用的回收方法4.1 什么是分代回收4.2 哪些对…

【Linux】:信号的产生

信号 一.前台进程和后台进程1.前台进程2。后台进程3.总结 二.自定义信号动作接口三.信号的产生1.键盘组合键2.kill信号进程pid3.系统调用1.kill函数2.raise函数3.abort函数 四.异常五.软件条件六.通过终端按键产生信号 一.前台进程和后台进程 1.前台进程 一个简单的代码演示 …

跟着chatgpt学习|1.spark入门

首先先让chatgpt帮我规划学习路径&#xff0c;使用Markdown格式返回&#xff0c;并转成思维导图的形式 目录 目录 1. 了解spark 1.1 Spark的概念 1.2 Spark的架构 1.3 Spark的基本功能 2.spark中的数据抽象和操作方式 2.1.RDD&#xff08;弹性分布式数据集&#xff09; 2…

JAVA时间常用操作工具类

小刘整理了JAVA中对时间的常用操作&#xff0c;封装了几种方法&#xff0c;简单方便&#xff0c;开箱即用。时间转字符串格式&#xff0c;字符串转时间&#xff0c;以及过去和未来的日期。除此之外&#xff0c;还新增了时间戳之差计算时分秒天的具体方案。 public static void …

【力扣:1707 1803】0-1字典树

思路&#xff1a;树上每个节点存储拥有该节点的数组元素的最小值&#xff0c;left节点表示0&#xff0c;right节点表示1&#xff0c;构建完成后遍历树当子节点没有比mi小的元素时直接输出-1&#xff0c;否则向下构造。 struct tree{int m;tree*leftnullptr,*rightnullptr;tree…

智能优化算法应用:基于海鸥算法无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用&#xff1a;基于海鸥算法无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用&#xff1a;基于海鸥算法无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.海鸥算法4.实验参数设定5.算法结果6.参考文献7.MATLAB…