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

news2024/11/28 0:29:22

目录

服务器端

代码

客户端

代码解析

服务器端

原理

遇到的阻碍以及解决办法

客户端

原理

遇到的阻碍以及解决办法

运行结果截图

总结


服务器端

代码

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

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

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

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

4D Gaussian Splatting:用于实时的动态场景渲染

Wu G, Yi T, Fang J, et al. 4d gaussian splatting for real-time dynamic scene rendering[J]. arXiv preprint arXiv:2310.08528, 2023. 更多参考资料如下&#xff1a; 文章总结&#xff1a;4D Gaussian Splatting for Real-Time Dynamic Scene Rendering&#xff1b;疑难问…

Java作业四

要求&#xff1a;编写带图形界面的聊天程序&#xff0c;实现让客户可以持续地发送消息给服务器&#xff0c;服务器也可以即时看到客户端发送的消息&#xff0c;并回消息给客户端。 程序运行界面如下&#xff1a; 老师写的&#xff1a; 1、客户端 package Demo02; import java.…

VM虚拟机中Ubuntu14.04安装VM tools后仍不能全屏显示

1、查看Ubuntu所支持的分辨率大小。 在终端处输入&#xff1a; xrandr&#xff0c;回车 2、输入你想设置的分辨率参数。 我设置的为1360x768&#xff0c;大家可以根据自己的具体设备设置。 在终端输入&#xff1a;xrandr -s 1360x768 注意&#xff1a;这里1360后边是字母 x 且…

C语言-内存函数详解

文章目录 1. memcpy使用和模拟实现2. memmove使用和模拟实现3. memset函数的使用4. memcmp函数的使用 1. memcpy使用和模拟实现 返回类型和参数&#xff1a; void * memcpy ( void * destination, const void * source, size_t num );1.函数memcpy从source的位置开始向后复制…

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

C/C++等级考试(1~8级)全部真题・点这里 第1题:数字字符求和 请编写一个程序实现以下功能:从一个字符串中,提取出所有的数字字符即0-9,并作为数求和。 时间限制:1000 内存限制:65536输入 一行字符串,长度不超过100,字符串中不含空格。输出 字符串中所有数字字符作为数…

【Mybatis】预编译/即时sql 数据库连接池

回顾 Mybatis是一个持久层框架.有两种方式(这两种方式可以共存) 1.注解 2.xml 一.传递参数 以使用#{} 来接受参数为例 (以上两种方式一样适用的) 1)传递单个参数 #{} 可以为任意名称 2)多个参数 默认的参数名称就是接口方法声明的形参 3)参数为对象 默认给每个对象的每个属性都…

探索 Linux vim/vi 编辑器:介绍、模式以及基本操作演示

&#x1f490;作者&#xff1a;insist-- &#x1f490;个人主页&#xff1a;insist-- 的个人主页 理想主义的花&#xff0c;最终会盛开在浪漫主义的土壤里&#xff0c;我们的热情永远不会熄灭&#xff0c;在现实平凡中&#xff0c;我们终将上岸&#xff0c;阳光万里 ❤️欢迎点…

笔记(三)maxflow push relabel与图像分割

笔记&#xff08;三&#xff09;maxflow push relabel与图像分割 1. Push-Relabel算法思想2.Push-Relabel算法原理示意图3.Push-Relabel算法具体实例4. push relabel与图割 1. Push-Relabel算法思想 对于一个网络流图: 该算法直观可以这样理解&#xff0c;先在源节点处加入充足…

2018年1月22日 Go生态洞察:Hello, 中国!

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

Java毕业设计 SpringBoot 车辆充电桩系统

Java毕业设计 SpringBoot 车辆充电桩系统 SpringBoot 车辆充电桩系统 功能介绍 首页 图片轮播 登录注册 充电桩展示 搜索充电桩 充电桩报修 充电常识 个人中心 我的收藏 后台管理 登录 首页 个人中心 维修员管理 用户管理 电桩类别管理 充电桩管理 充电桩报修管理 维修回复管…

DBS note4:Buffer Management

目录 1、介绍 2、缓冲池 3、处理页面请求 4、LRU替换和时钟策略 1&#xff09;顺序扫描性能 - LRU 5、最近最常使用替换策略&#xff08;MRU Replacement&#xff09; 1&#xff09;Sequential Scanning Performance - MRU 6、练习题 1&#xff09;判断真假 2&#xf…

HCIP-十、BGP基础

十、BGP基础 实验拓扑实验需求及解法1.R1 属于 AS100&#xff0c;R2/3/4 属于 AS200&#xff0c;R5 属于 AS3002.AS200 内运行 OSPF3.建立 IBGP 邻居4.建立 EBGP 邻居5.BGP 发布路由6.路由黑洞 实验拓扑 实验需求及解法 本实验模拟 ISP 网络拓扑&#xff0c;运行 BGP。如图所示…

python中模块的创建及引用(import as,import,from)

模块&#xff08;module&#xff09;简介&#xff1a; 1.模块化&#xff0c;模块化指将一个完整的程序分解为一个一个的小模块&#xff0c; 通过将模块组合&#xff0c;来搭建出一个完整的程序 2.不采用模块化就是统一将所有的代码编写到一个文件中&#xff0c;采用 模块化就是…

chrome driver 截图和填表

昨天突然有一个需求&#xff08;自己的&#xff09;&#xff0c;想把某个网站题目主体部分翻译并保存成图片&#xff0c;开始时用了国内网站的翻译&#xff08;人工、简单翻译&#xff09;&#xff0c;后来发现很多地方翻译的不尽人意&#xff0c;于是只好用翻译插件对原始网站…

【H5 Canvas】【平面几何】特殊图形绘制(【带旋转】箭头/正多边形/正多尖角形等)

文章目录 直线/弧线 箭头 直线/弧线 箭头 // startX,startY 起始坐标 // endX,endY 结束坐标 // radian 圆弧角度,取值[0&#xff0c;PI]; 0表示画直线箭头&#xff0c;否则画圆弧箭头 CanvasRenderingContext2D.prototype.drawArrow function(startX,startY,endX,endY,radia…

【Spring篇】spring核心——AOP面向切面编程

目录 想要彻底理解AOP&#xff0c;我觉得你的先要了解框架的模块化思想&#xff0c;为此先记录框架在讲AOP 什么是java框架&#xff1f;为什么要出现框架&#xff1f; 我总结以下七点来讲述和帮助理解java框架思想 什么是AOP&#xff1f; 如何理解上面这句话呢&#xff1…

HarmonyOS安装三方库遇到的问题

使用开发电脑系统为&#xff1a;MacOS, 开发工具为&#xff1a;DevEco-Studio版本号3.1.1 Release。在控制栏使用终端工具输入命令&#xff1a;ohpm install ohos/lottie遇到的第一个问题如下图。 解决方案&#xff1a; 1、在首选项中找到ohpm的安装路径。 2、打开bash_profil…

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

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

pyhon数据分析A股股票策略实际买卖总结(每月末更新数据)

简介 本篇文章主要记录python数据分析a股股票选股后实际买卖的记录。 选股策略 低位寻股&#xff0c;筛选出低位股价股票已经做过调整的股票&#xff0c;做短线交易&#xff08;不超过7天&#xff09;&#xff0c;不贪&#xff0c;小赚即走。分三个时段&#xff0c;开盘三十…