网络编程Day9_IO多路复用 20240821

news2025/1/11 12:55:10

运行1个服务器和2个客户端实现效果:

服务器和2个客户端互相聊天,服务器和客户端都需要使用select模型去实现

服务器要监视2个客户端是否连接,2个客户端是否发来消息以及服务器自己的标准输入流

客户端要监视服务器是否发来消息以及客户端自己的标准输入流

在不开线程的情况下,实现互相聊天

select_server.c

#include <myhead.h>

typedef struct Msg
{
    int fd;
    char text[128];
} Msg;

// 将新连接生成套接字描述符添加进客户端套接字描述符数组
void insert_client(int *client_arr, int *client_len, int newfd)
{
    client_arr[*client_len] = newfd; // 将新描述符添加到数组
    (*client_len)++;                 // 增加客户端数量
}

// 查找客户端描述符位置下标
int find_client(int *client_arr, int client_len, int newfd)
{
    for (int i = 0; i < client_len; i++)
    {
        if (client_arr[i] == newfd) // 如果找到对应的描述符
        {
            return i; // 返回下标
        }
    }

    return -1; // 未找到,返回-1
}

void remove_client(int *client_arr, int *client_len, int newfd)
{
    int tar = find_client(client_arr, *client_len, newfd); // 查找描述符位置
    if (tar == -1)                                         // 如果未找到
    {
        return; // 直接返回
    }
    int i = -1;
    for (i = tar; i < *client_len - 1; i++)
    {
        client_arr[i] = client_arr[i + 1]; // 移动数组元素
    }
    client_arr[i] = 0; // 清空最后一个元素
    (*client_len)--;   // 减少客户端数量
}

int main(int argc, const char *argv[])
{
    if (argc != 3) // 检查参数数量
    {
        printf("请输入正确的IP地址和端口号\n");
        return -1; // 参数错误,返回-1
    }

    // 1、创建套接字,设置端口号快速重启
    int sfd = socket(AF_INET, SOCK_STREAM, 0); // 创建套接字
    if (sfd == -1)                             // 检查套接字创建是否成功
    {
        perror("socket error");
        return -1; // 创建失败,返回-1
    }
    int opt = -1;
    if (setsockopt(sfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt)) == -1) // 设置套接字选项
    {
        perror("setsockopt error");
        return -1; // 设置失败,返回-1
    }

    // 2、初始化地址信息并绑定
    struct sockaddr_in sin;                                    // 定义地址结构
    sin.sin_family = AF_INET;                                  // IPv4
    sin.sin_addr.s_addr = inet_addr(argv[1]);                  // 设置IP地址
    sin.sin_port = htons(atoi(argv[2]));                       // 设置端口号
    if (bind(sfd, (struct sockaddr *)&sin, sizeof(sin)) == -1) // 绑定套接字
    {
        perror("bind error");
        return -1; // 绑定失败,返回-1
    }

    // 3.监听端口连接
    if (listen(sfd, 128) == -1) // 开始监听
    {
        perror("listen error");
        return -1; // 监听失败,返回-1
    }

    // 4.接收连接请求,创建连接
    fd_set readfds;        // 定义文件描述符集合
    FD_ZERO(&readfds);     // 清空集合
    FD_SET(sfd, &readfds); // 将监听套接字添加到集合
    FD_SET(STDIN_FILENO, &readfds);
    int client_arr[128] = {0}; // 客户端描述符数组
    int client_len = 0;        // 当前客户端数量

    Msg msg = {0};

    while (1) // 主循环
    {
        fd_set temp = readfds;              // 复制集合
        select(FD_SETSIZE, &temp, 0, 0, 0); // 监视文件描述符
        if (FD_ISSET(sfd, &temp))           // 如果有新连接
        {
            bzero(&msg, sizeof(msg));
            struct sockaddr_in cin;                                     // 定义客户端地址结构
            socklen_t cin_len = sizeof(cin);                            // 地址结构大小
            int newfd = accept(sfd, (struct sockaddr *)&cin, &cin_len); // 接受连接
            if (newfd == -1)                                            // 检查连接是否成功
            {
                perror("accept error");
                return -1; // 接受失败,返回-1
            }
            printf("新客户端[%d]连接\n", newfd); // 输出连接信息

            msg.fd = newfd;
            strcpy(msg.text, "已上线");
            for (int j = 0; j < client_len; j++)
            {
                int cfd = client_arr[j];
                if (cfd != newfd && cfd != sfd)
                {
                    send(cfd, &msg, sizeof(msg), 0);
                }
            }

            FD_SET(newfd, &readfds);                       // 将新描述符添加到集合
            insert_client(client_arr, &client_len, newfd); // 添加到客户端数组
        }
        else // 处理已连接的客户端
        {
            for (int i = 0; i < client_len; i++)
            {
                int newfd = client_arr[i];  // 获取客户端描述符
                if (FD_ISSET(newfd, &temp)) // 如果有数据可读
                {
                    char buf[128] = "";                      // 缓冲区
                    int res = read(newfd, buf, sizeof(buf)); // 读取数据
                    if (res == 0)                            // 如果客户端断开连接
                    {
                        msg.fd = newfd;
                        strcpy(msg.text, "已下线");
                        printf("客户端[%d]断开连接\n", msg.fd);
                        for (int j = 0; j < client_len; j++)
                        {
                            int cfd = client_arr[j];
                            if (cfd != newfd && cfd != sfd)
                            {
                                send(cfd, &msg, sizeof(msg), 0);
                            }
                        }

                        FD_CLR(newfd, &readfds);                       // 从集合中移除
                        remove_client(client_arr, &client_len, newfd); // 从数组中移除
                        close(newfd); // 关闭描述符
                        break;        // 退出当前循环
                    }
                    printf("客户端[%d]发来消息:%s\n", newfd, buf); // 输出客户端消息

                    msg.fd = newfd;
                    strcpy(msg.text, buf);
                    for (int j = 0; j < client_len; j++)
                    {
                        int cfd = client_arr[j];
                        if (cfd != newfd && cfd != sfd)
                        {
                            send(cfd, &msg, sizeof(msg), 0);
                        }
                    }
                }
                
            }
        }

        if (FD_ISSET(STDIN_FILENO, &temp))
                {
                    bzero(&msg, sizeof(msg));
                    msg.fd = 0;
                    scanf("%s", msg.text);
                    while (getchar() != 10);
                    // 将数据发送给客户端
                    for (int j = 0; j < client_len; j++)
                    {
                        int cfd = client_arr[j];
                        if (cfd != sfd)
                        {
                            send(cfd, &msg, sizeof(msg), 0);
                        }
                    }
                }
    }

    return 0; // 程序结束
}

select_client.c

#include <myhead.h>
#define SER_PORT 6666		   // 服务器端口号
#define SER_IP "192.168.0.222" // 服务器IP地址

typedef struct Msg
{
	int fd;
	char text[128];
} Msg;

int main(int argc, const char *argv[])
{
	if (argc != 3)
	{
		printf("请输入正确的IP地址和端口号\n");
		return -1;
	}

	// 1、创建用于通信的套接字文件描述符
	int cfd = socket(AF_INET, SOCK_STREAM, 0);
	if (cfd == -1)
	{
		perror("socket error");
		return -1;
	}
	printf("cfd = %d\n", cfd);
	// 将端口号快速重用
	int reuse = 1;
	if (setsockopt(cfd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse)) == -1)
	{
		perror("setsockopt error");
		return -1;
	}
	printf("端口号快速重用成功\n");

	// 2、绑定IP地址和端口号
	// 2.1 填充地址信息结构体
	struct sockaddr_in cin;
	cin.sin_family = AF_INET;
	cin.sin_addr.s_addr = inet_addr(argv[1]);
	cin.sin_port = htons(atoi(argv[2]));

	// 2.2 绑定工作
	if (bind(cfd, (struct sockaddr *)&cin, sizeof(cin)) == -1)
	{
		perror("bind error");
		return -1;
	}
	printf("bind success\n");

	// 3、连接到服务器
	// 3.1 填充服务器地址信息结构体
	struct sockaddr_in sin;
	sin.sin_family = AF_INET;
	sin.sin_port = htons(SER_PORT);
	sin.sin_addr.s_addr = inet_addr(SER_IP);

	// 3.2 连接服务器
	if (connect(cfd, (struct sockaddr *)&sin, sizeof(sin)) == -1)
	{
		perror("connect error");
		return -1;
	}
	printf("服务器连接成功\n");

	// 4、数据收发
	char buf[128] = "";
	fd_set readfds;					// 定义文件描述符集合
	FD_ZERO(&readfds);				// 清空集合
	FD_SET(cfd, &readfds);			// 将监听套接字添加到集合
	FD_SET(STDIN_FILENO, &readfds); // 将标准输入添加到集合

	Msg msg = {0};

	while (1)
	{
		bzero(buf, sizeof(buf));
		bzero(&msg, sizeof(msg));
		fd_set temp = readfds;				// 复制集合
		select(FD_SETSIZE, &temp, 0, 0, 0); // 监视文件描述符
		if (FD_ISSET(cfd, &temp))
		{
			int res = recv(cfd, &msg, sizeof(msg), 0);
			if (res == -1)
			{
				perror("revc error");
			}
			else if(res == 0)
			{
				printf("服务器已下线\n");
				break;
			}
			if (msg.fd == 0)
			{
				printf("服务器发来消息: %s\n",msg.text);
			}
			else
			{
				printf("客户端[%d]发来消息: %s\n", msg.fd, msg.text);
			}
		}
		if (FD_ISSET(STDIN_FILENO, &temp))
		{
			bzero(buf, sizeof(buf));
			fgets(buf, sizeof(buf), stdin);
			buf[strlen(buf) - 1] = 0;
			if (strcmp(buf, "quit") == 0)
			{
				printf("退出成功\n");
				break;
			}
			// 将数据发送给服务器
			send(cfd, buf, strlen(buf), 0);
		}
	}

	// 5、关闭套接字
	close(cfd);

	return 0;
}

运行效果

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

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

相关文章

Go小技巧易错点100例(十七)

Go定时任务 在Go语言中&#xff0c;定时任务&#xff08;也称为定时器或cron作业&#xff09;具有多种作用&#xff0c;这些作用在应用程序的开发和运维中非常有用。以下是一些常见使用场景&#xff1a; 任务调度&#xff1a;定时任务可以在特定的时间点执行特定的任务&#…

云渲染服务大揭秘:为何它值得成为你的渲染新选择

云渲染是一种基于云计算的服务&#xff0c;它利用大量高性能计算机组成的集群来渲染高质量的图像和动画。这种服务能够显著加快3D动画或视觉效果项目的渲染速度&#xff0c;将原本可能需要数天的渲染任务缩短至数小时。 云渲染的优势 本文将探讨使用云渲染的好处&#xff0c;并…

【ACM出版,高录用EI快检索】第七届计算机信息科学与人工智能国际学术会议(CISAI 2024,9月6-8)

第七届计算机信息科学与人工智能国际学术会议(CISAI 2024) 将于2024年09月6-8日在中国浙江-绍兴举行。 计算机信息科学与人工智能国际学术会议的主题主要围绕“信息科学”与“人工智能”的最新研究展开&#xff0c;旨在荟聚世界各地该领域的专家、学者、研究人员及相关从业人员…

React学习笔记,从入门到砸门

项目构建命令 npx create-react-app react-basic npx&#xff1a;node语法 create-react-app&#xff1a;项目模板 react-basic&#xff1a;项目名称 项目结构 项目打包和本地预览 项目打包npm run build本地预览&#xff08;模拟服务器运行项目&#xff09; 安装本地服务包…

STM32f407 网络接收 fpga 的 bin 文件并更新到 fpga series7(3)

STM32f407 网络接收 fpga 的 bin 文件并更新到 fpga series7(3) 简介 实验 3&#xff1a;在搭建好 tcp 服务器&#xff0c;并拟定好协议的前提下&#xff0c;接收每一个 bin 文件的块&#xff0c;配置到 fpga。 原理图 fpga fpga1 stm32 接线总结 // fpga引脚 stm32…

快速了解矿用电源特性及其性能测试利器电源ate检测系统

在矿产资源开采的每一个环节&#xff0c;矿用电源都扮演着幕后英雄的角色&#xff0c;它的作用不可小觑。那么什么是矿用电源呢&#xff1f;电源ate检测系统如何助力矿用电源性能测试呢&#xff1f; 矿用电源模块介绍 矿用电源是专门用于矿井等地下作业场所的重要电源设备&…

阿里MAXCOMPUTE数据专辑信息读取并同步数据表

阿里MAXCOMPUTE数据专辑信息读取并同步数据表 在阿里云大数据体系中&#xff0c;我们可以使用数据地图的数据专辑&#xff0c;对数据的类别等进行一个管理 那么管理后的数据&#xff0c;我们想要落表进行相关的数据分析&#xff0c;如何做呢&#xff1f; 查看阿里云官方文档…

虚幻5|制作刀光粒子效果

一&#xff0c;创建一个粒子效果 1.Niagara系统 2.右键添加发射器&#xff0c;创建一个空白 3.点击空白的渲染&#xff0c;选择条带渲染器 4.右侧选择自定义侧面矢量 5.按顺序如下&#xff0c;编辑刀光的周期和方向 6.添加一个spawn per frame&#xff0c;使刀光每帧都在生成&…

Upload-Lab第13关:POST上传方式如何巧妙利用%00截断法绕过上传验证

第13关概述 在Upload-Lab第13关中&#xff0c;服务器会对上传的文件进行严格的扩展名检查。只有符合白名单的扩展名&#xff08;如.jpg、.png等&#xff09;才能成功上传。我们的目标是绕过这种检查&#xff0c;将恶意文件&#xff08;如.php&#xff09;上传到服务器。以下是…

图神经网络教程4-卷积图神经网络

介绍 卷积神经网络在涉及图像的预测任务上取得了最先进的性能。通过将权值学习核与输入图像卷积&#xff0c;CNN根据其视觉外观提取感兴趣的特征&#xff0c;无论它们在图像中的位置是哪里。虽然图像只是图的一个特殊情况(见图1 (a))&#xff0c;但是为图领域定义一个广义卷积…

了解同步带选择同步带

同步带和轮选型 同步带传动属于皮带传动&#xff0c;但是改进了传统皮带传动无法保持严格的传动比的打滑问题&#xff0c;传统皮带传动依靠皮带和皮带轮张紧时产生的摩擦力传输动力&#xff0c;但是从动轮遇到障碍或超载荷时&#xff0c;皮带会在皮带轮产生滑动。 解决打滑问题…

企业高性能web服务器【Nginx详解】

一.Web 服务基础介绍 1.1 互联网发展历程 1993年3月2日&#xff0c;中国科学院高能物理研究所租用AT&T公司的国际卫星信道建立的接入美国SLAC国家实 验室的64K专线正式开通&#xff0c;成为我国连入Internet的第一根专线。 1995年马云开始创业并推出了一个web网站 中国黄页…

【其它-高效处理小技巧】如何批量备份263企业邮箱邮件

如何批量备份263企业邮箱邮件 近期由于有人离职&#xff0c;邮箱要注销&#xff0c;之前邮箱内有5000多封沟通邮件&#xff0c;为避免将来找不到沟通过程&#xff0c;所以需要备份。 目的&#xff1a;一次性备份所有沟通邮件 方法一&#xff1a; 少于20封邮件&#xff0c;推荐…

基于vue框架的爱心公益网站532y9(程序+源码+数据库+调试部署+开发环境)系统界面在最后面。

系统程序文件列表 项目功能&#xff1a;用户,志愿者,公益资讯,捐赠物资,公益项目,项目报名,公益类型 开题报告内容 基于Vue框架的爱心公益网站 开题报告 一、项目背景与意义 在快速发展的现代社会中&#xff0c;公益事业作为社会文明进步的重要标志&#xff0c;越来越受到…

创建GPTs,打造你的专属AI聊天机器人

在2023年11月的「OpenAI Devday」大会上&#xff0c;OpenAI再度带来了一系列令人瞩目的新功能&#xff0c;其中ChatGPT方面的突破尤为引人关注。而GPTs的亮相&#xff0c;不仅标志着个性化AI时代的到来&#xff0c;更为开发者和普通用户提供了前所未有的便利。接下来&#xff0…

WPS又崩了,在黑神话中挤出一条热搜!

吉祥知识星球http://mp.weixin.qq.com/s?__bizMzkwNjY1Mzc0Nw&mid2247485367&idx1&sn837891059c360ad60db7e9ac980a3321&chksmc0e47eebf793f7fdb8fcd7eed8ce29160cf79ba303b59858ba3a6660c6dac536774afb2a6330#rd 《网安面试指南》http://mp.weixin.qq.com/s?…

idea中如何不重启tomcat 即可看到修改内容变化

我 | 在这里 ⭐ 全栈开发攻城狮、全网10W粉丝、2022博客之星后端领域Top1、专家博主。 &#x1f393;擅长 指导毕设 | 论文指导 | 系统开发 | 毕业答辩 | 系统讲解等。已指导60位同学顺利毕业。持续接单中。。。 ✈️个人公众号&#xff1a;热爱技术的小郑。回复 Java全套视频教…

基于ElementPlus的分页表格组件ReTable

分页表格ReTable 组件实现基于 Vue3 Element Plus Typescript&#xff0c;同时引用 vueUse lodash-es tailwindCss (不影响功能&#xff0c;可忽略) 基于ElTable和ElPagination组件封装的分页表格&#xff0c;支持本地分页以及远程请求两种方式。本地数据分页自带全量数据的…

QT聊天室基于Tcp

server.cpp #include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget),server(new QTcpServer(this)) // 给服务器指针对象实例化空间{ui->setupUi(this); }Widget::~Widget() {delete ui; }…

集团数字化转型方案(一)

集团数字化转型方案通过系统集成先进的物联网&#xff08;IoT&#xff09;、大数据分析、人工智能&#xff08;AI&#xff09;和云计算技术&#xff0c;构建一个全面智能化的运营生态系统&#xff0c;涵盖从数据驱动的决策支持、智能化业务流程优化、到全渠道客户体验提升的各个…