使用select

客户端

服务端

     1	#include<myhead.h>
     2	 
     3	#define SER_PORT 6666       //服务器端口
     4	#define SER_IP "127.0.0.1"  //服务器ip
     5	 
     6	 
     7	int main(int argc, const char *argv[])
     8	{
     9		//创建套接字
    10		int sfd=socket(AF_INET,SOCK_STREAM,0);
    11		if(sfd==-1)
    12		{
    13			perror("socket error");
    14			return -1;
    15		}
    16	 
    17		//绑定 地址和端口号
    18		struct sockaddr_in sin;
    19		sin.sin_family=AF_INET; 
    20		sin.sin_port=htons(SER_PORT); 
    21		sin.sin_addr.s_addr=inet_addr(SER_IP); 
    22		if(bind(sfd,(struct sockaddr*)&sin,sizeof(sin))==-1)
    23		{
    24			perror("bind error");
    25			return -1;
    26		}
    27	
    28	
    29		//定义用户套接字数组 个数
    30		int client_arr[100]={0};
    31		int client_count=0;
    32	
    33		//开启监听
    34		if(listen(sfd,128)==-1)
    35		{
    36			perror("listen error");
    37			return -1;
    38		}
    39	
    40		//定义文件描述符集
    41		fd_set readfds;
    42		FD_ZERO(&readfds);
    43	 
    44	
    45		//添加
    46		FD_SET(0,&readfds);
    47		FD_SET(sfd,&readfds);
    48	
    49		//最大的文件描述符
    50		int maxfd=sfd;
    51	
    52		//存储客户端地址信息结构体数组
    53		struct sockaddr_in cin_arr[128];
    54	 
    55		int newfd=-1;
    56	
    57		struct sockaddr_in cin;        //接收客户端的信息
    58		socklen_t addrlen =sizeof(cin); //用于接收长度
    59	
    60		char wbuf[128]=""; // 信息搬运工
    61		while(1)
    62		{
    63			fd_set temp =readfds;
    64	
    65			//判断是否有文件描述符被激活
    66			int res=select(FD_SETSIZE,&temp,0,0,0);
    67			if(res==-1)
    68			{
    69				perror("select error");
    70				return -1;
    71			}
    72	
    73			//当程序执行自此表示有文件描述符被激活
    74			for(int i=0;i<=maxfd;i++)        //遍历文件描述符
    75			{
    76				//查找是什么事件触发了文件描述符被激活
    77				int res=FD_ISSET(i,&temp);
    78				if(res==0)
    79				{
    80					continue;
    81				}
    82	
    83				//程序执行至此表示i是被触发事件的文件描述符
    84				//判断sfd是否触发
    85				if(i==sfd)
    86				{
    87					//接收连接请求
    88					int newfd=accept(sfd,(struct sockaddr*)&cin,&addrlen);
    89					if(newfd==-1)
    90					{
    91						perror("accept error");
    92						return -1;
    93					}
    94					printf("成功连接到一个客户端\n");
    95	
    96					//将新连接的客户端加入监视列表,下一次循环,碰到select就能监视到
    97					FD_SET(newfd,&readfds);
    98					//将客户端地址信息结构体放入客户端地址信息数组中
    99					cin_arr[newfd]=cin;
   100					//更新最大文件描述符
   101					if(maxfd<newfd)
   102					{
   103						maxfd=newfd;
   104					}
   105				}
   106				//判断是否是标准输入流触发该事件
   107				else if(0==i)
   108				{
   109				
   110					fgets(wbuf,sizeof(wbuf),stdin);   //从终端读取消息
   111					wbuf[strlen(wbuf)-1]=0;
   112					printf("发送的消息>>>%s\n",wbuf);
   113	
   114					//将该消息发送给所有客户端
   115					for(int i=4;i<=maxfd;i++)
   116					{
   117						if(send(i,wbuf,strlen(wbuf),0)==-1)
   118						{
   119							perror("send error");
   120							return -1;
   121						}
   122					}
   123					printf("发送成功\n");
   124				}
   125				else
   126				{
   127	
   128					char buf[128]="";
   129	
   130					//清空容器
   131					bzero(buf,sizeof(buf));
   132					//从客户端套接字中读取数据
   133					int res=recv(i,buf,sizeof(buf),0);
   134					if(res==-1)
   135					{
   136						perror("recv error");
   137						return -1;
   138					}
   139	
   140					else if(res==0)
   141					{
   142						printf("用户已经下线\n");
   143						close(i);
   144	
   145						FD_CLR(i,&readfds);      //删除
   146						//更新最大文件描述符
   147						for(int j=maxfd;j>=sfd;j--)
   148						{
   149							if(FD_ISSET(j,&readfds))
   150							{
   151								maxfd =j;
   152								break;
   153							}
   154						}
   155						continue;
   156					}
   157					//正常收到客户端发来的消息
   158					printf("[%s:%d]:%s\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),buf);
   159				}
   160			}
   161		}
   162		//关闭
   163		close(sfd);
   164	 
   165		return 0;
   166	}

客户端

     1	#include<myhead.h>
     2	typedef struct sockaddr_in addr_in_t;
     3	typedef struct sockaddr addr_t;
     4	typedef struct sockaddr_un addr_un_t;
     5	int main(int argc, const char *argv[])
     6	{
     7		if(argc!=2)
     8		{
     9			printf("请输入正确的端口号\n");
    10			return -1;
    11		}
    12	
    13		int port = atoi(argv[1]);
    14	
    15		//创建描述符
    16		int client = socket(AF_INET,SOCK_STREAM,0);
    17	
    18		//绑定连接服务器
    19		addr_in_t addr;
    20		addr.sin_family=AF_INET;
    21		addr.sin_port=htons(port);
    22		addr.sin_addr.s_addr=inet_addr("127.0.0.1");
    23		if(connect(client,(addr_t*)&addr,sizeof(addr))==-1)
    24		{
    25			perror("connect error");
    26			return -1;
    27		}
    28	
    29		fd_set readfds;  //准备描述符集合
    30		FD_ZERO(&readfds);  //初始化
    31	
    32		FD_SET(client,&readfds);  //将客户端文件描述符加入描述符集合
    33		FD_SET(0,&readfds);  //将标准输入流添加入描述人集合
    34	
    35	
    36		char buf[1024]="";
    37		while(1)
    38		{
    39			//定义一个描述符集合
    40			fd_set temp=readfds;       
    41			select(FD_SETSIZE,&temp,0,0,NULL);       //阻塞
    42	
    43			if(FD_ISSET(client,&temp))
    44			{
    45				//清空
    46				bzero(buf,sizeof(buf));
    47				recv(client,buf,sizeof(buf),0);
    48				printf("接收消息:%s\n",buf);
    49			}
    50	
    51			if(FD_ISSET(0,&temp))
    52			{
    53				printf("请输入>>>");
    54				scanf("%s",buf);
    55				if(strcmp(buf,"q")==0)
    56				{
    57					break;
    58				}
    59	
    60				//发送数据
    61				send(client,buf,strlen(buf),0);
    62	
    63			}
    64		}
    65	
    66		return 0;
    67	}

思维导图

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

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

相关文章

开源大模型LLaMA架构介绍

大模型相关目录 大模型&#xff0c;包括部署微调prompt/Agent应用开发、知识库增强、数据库增强、知识图谱增强、自然语言处理、多模态等大模型应用开发内容 从0起步&#xff0c;扬帆起航。 swift与Internvl下的多模态大模型分布式微调指南&#xff08;附代码和数据&#xff…

思科设备静态路由实验

拓扑及需求 网络拓扑及 IP 编址如图所示&#xff1b;PC1 及 PC2 使用路由器模拟&#xff1b;在 R1、R2、R3 上配置静态路由&#xff0c;保证全网可达&#xff1b;在 R1、R3 上删掉上一步配置的静态路由&#xff0c;改用默认路由&#xff0c;仍然要求全网可达。 各设备具体配置…

前端技巧——复杂表格在html当中的实现

应用场景 有时候我们的表格比较复杂&#xff0c;表头可能到处割裂&#xff0c;我们还需要写代码去完成这个样式&#xff0c;所以学会在原生html处理复杂的表格还是比较重要的。 下面我们来看这一张图&#xff1a; 我们可以看到有些表头项的规格不太一样&#xff0c;有1*1 2*…

Unity Protobuf3.21.12 GC 问题(反序列化)

背景&#xff1a;Unity接入的是 Google Protobuf 3.21.12 版本&#xff0c;排查下来反序列化过程中的一些GC点&#xff0c;处理了几个严重的&#xff0c;网上也有一些分析&#xff0c;这里就不一一展开&#xff0c;默认读者已经略知一二了。 如果下面有任何问题请评论区留言提…

实现 FastCGI

CGI的由来&#xff1a; 最早的 Web 服务器只能简单地响应浏览器发来的 HTTP 请求&#xff0c;并将存储在服务器上的 HTML 文件返回给浏 览器&#xff0c;也就是静态 html 文件&#xff0c;但是后期随着网站功能增多网站开发也越来越复杂&#xff0c;以至于出现动态技 术&…

2020 位示图

2020年网络规划设计师上午真题解析36-40_哔哩哔哩_bilibili 假设某计算机的字长为32位&#xff0c;该计算机文件管理系统磁盘空间管理采用位示图&#xff08;bitmap&#xff09;&#xff0c;记录磁盘的使用情况。若磁盘的容量为300GB&#xff0c;物理块的大小为4MB&#xff0c;…

【网络安全】漏洞挖掘:IDOR实例

未经许可&#xff0c;不得转载。 文章目录 正文 正文 某提交系统&#xff0c;可以选择打印或下载passport。 点击Documents > Download后&#xff0c;应用程序将执行 HTTP GET 请求&#xff1a; /production/api/v1/attachment?id4550381&enamemId123888id为文件id&am…

C语言 | Leetcode C语言题解之第354题俄罗斯套娃信封问题

题目&#xff1a; 题解&#xff1a; int cmp(int** a, int** b) {return (*a)[0] (*b)[0] ? (*b)[1] - (*a)[1] : (*a)[0] - (*b)[0]; }int maxEnvelopes(int** envelopes, int envelopesSize, int* envelopesColSize) {if (envelopesSize 0) {return 0;}qsort(envelopes, …

JVM 有哪些垃圾回收器?

JVM 有哪些垃圾回收器&#xff1f; 图中展示了7种作用于不同分代的收集器&#xff0c;如果两个收集器之间存在连线&#xff0c;则说明它们可以搭配使用。虚拟机所处的区域则表示它是属于新生代还是老年代收集器。 新生代收集器&#xff08;全部的都是复制算法&#xff09;&…

wps题注为表格或图片编号

word中为表格添加题注&#xff1a; 问题&#xff1a;多次或多人编辑导致--序号不能联动更新&#xff08;域代码不一致,如图&#xff09; 所以是否可以批量替换word里的域代码&#xff1f;如果可以这问题就解决了————失败 解决办法&#xff1a; 如图&#xff0c;复制表头&…

协处理器+流水线 (9)

3级流水线 流程&#xff1a; 取指令 译码 执行。 每一个时钟周期都可以执行一个指令。 提高CPU的能力有两种方法&#xff0c; 1 提高时钟频率&#xff0c;造成单位时间内执行的指令更多。 2 减少每条指令的平均指令周期数CPI &#xff0c;CPI我不太懂&#xff0c;但大概的…

2024.8.21 作业

一个服务器和两个客户端聊天 代码&#xff1a; /*******************************************/ 文件名&#xff1a;server.c /*******************************************/ #include <myhead.h> #define SER_IP "192.168.2.7" // 服务器IP #define SER…

C#开发基础之100个常用的C#正则表达式

前言 正则表达式是处理字符串的强大工具&#xff0c;特别是在文本搜索、替换和验证中。本文将100个常用的C#正则表达式进行分类&#xff0c;以帮助我们更快速地找到适合的正则表达式解决方案。 1. 基础匹配 这些正则表达式用于匹配一些基本的字符或字符串模式。 匹配任意字…

Linux信号机制探析--信号的产生

&#x1f351;个人主页&#xff1a;Jupiter. &#x1f680; 所属专栏&#xff1a;Linux从入门到进阶 欢迎大家点赞收藏评论&#x1f60a; 目录 &#x1f4da;信号什么是信号&#xff1f;为什么要有信号&#xff1f;查看Linux系统中信号 &#x1f388;信号产生&#x1f4d5;kill…

【计算机网络】网络版本计算器

此前我们关于TCP协议一直写的都是直接recv或者read&#xff0c;有了字节流的概念后&#xff0c;我们知道这样直接读可能会出错&#xff0c;所以我们如何进行分割完整报文&#xff1f;这就需要报头来解决了&#xff01; 但当前我们先不谈这个话题&#xff0c;先从头开始。 将会…

Kubectl命令、初识pod、namespace

文章目录 一、Kubectl简介基础命令1.基本信息命令2.创建和更新资源命令3.删除资源命令4. 查看日志和调试命令5. 端口转发和复制文件命令6. 部署管理命令7. 伸缩命令8. 配置和上下文管理命令9.常用命令 二、Pod简介核心概念pod常见状态调度和初始化阶段容器创建和运行阶段异常状…

Zookeeper服务注册及心跳机制详解

ZooKeeper提供了一种类似于文件目录的结构来保存key值&#xff0c;其提供了四种key类型&#xff0c;分别是持久节点&#xff0c;临时节点&#xff0c;持久有序节点&#xff0c;临时有序节点。其中临时节点的特性是当创建此节点的会话断开时&#xff0c;节点也会被删除。这一特性…

Apache Commons-IO 库

Apache Commons-IO是Apache开源基金组织提供的一组有关IO&#xff08;Input/Output&#xff09;操作的小框架。这个库的主要目的是为了提高IO流的开发效率&#xff0c;减少在进行文件读写、目录遍历等操作时编写的样板代码量。通过使用Commons-IO库&#xff0c;开发者可以更加简…

WT32-ETH01开发板模块,启明云端物联网方案,乐鑫ESP32多样化开发应用

在物联网(IoT)的浪潮中&#xff0c;无线Wi-Fi模块作为连接传统硬件与现代智能网络的桥梁&#xff0c;正逐渐成为智能家居和设备通信不可或缺的一部分。Wi-Fi模块也被称为串口Wi-Fi模块&#xff0c;是一种嵌入式模块&#xff0c;它能够将串口或TTL电平信号转换为符合Wi-Fi无线网…