IO复用之select

news2025/4/6 6:14:22

目录

一.select方法介绍

2.1 select 系统调用的原型

2.2 集合的数据结构

2.2.1 fd_set 结构如下:

2.2.2 关于集合fd_set的解析

2.3 select第一个参数

2.4 select方法之超时时间timeout

2.5 select方法的用法简述及返回值

2.6 如何检测集合中有哪些描述符有事件就绪

三.select应用

3.1 select小实例

 3.2 结合tcp编程


一.select方法介绍

select 系统调用的用途是:在一段指定时间内,监听用户感兴趣的文件描述符的可读、可写和异常等事件。

2.1 select 系统调用的原型

select 成功时返回就绪(可读、可写和异常)文件描述符的总数。如果在超时时间内没有任何文件描述符就绪,select将返回 0。select 失败是返回-1.如果在 select 等待期间,程序接收到信号,则select 立即返回-1,并设置errno 为EINTR。

#include <sys/select.h>
int select(int maxfd, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, struct timeval *timeout);
  • maxfd 参数指定的被监听的文件描述符的总数。它通常被设置为 select 监听的所有文件描述符中的最大值+1。
  • readfds、 writefds 和 exceptfds 参数分别指向可读、可写和异常等事件对应的文件描述符集合。应用程序调用 select 函数时,通过这 3 个参数传入自己感兴趣的文件描述符。
  • select 返回时,内核将修改它们来通知应用程序哪些文件描述符已经就绪

2.2 集合的数据结构

select如何通知应用程序有哪些描述符数据就绪了,用户又是如何将这些描述符添加到select这个集合中?具体看下面关于集合fd_set的解析;

2.2.1 fd_set 结构如下:

#define __FD_SETSIZE 1024
typedef long int fd_mask;
#define __NFDBITS(8*(int)sizeof(__fd_mask))

typedef struct
{
    #ifdef __USE_XOPEN
    __fd_mask fds_bits[__FD_SETSIZE/__NFDBITS];//这是一个数组,依次是数组类型 数组名[数组个数]
    # define __FDS_BITS(set)((set)->fds_bits)
    #else
    __fd_mask __fds_bits[__FD_SETSIZE / __NFDBITS];
    # define __FDS_BITS(set)((set)->__fds_bits)
    #endif
}fd_set;

2.2.2 关于集合fd_set的解析

select如何通知应用程序有哪些描述符数据就绪了,用户又是如何将这些描述符添加到select这个集合中?我们首先来看关于集合fd_set的解析 :

也就是说这个结构体fd_set实际定义了1024个位。

通过下列宏可以访问 fd_set 结构中的位:

FD_ZERO(fd_set *fdset);// 清除 fdset 的所有位,就是将所有的位都置为0;

FD_SET(int fd,fd_set *fdset);// 设置 fdset 的位 fd,就是将某个描述符对应的位设置为1,就是将某个描述符添加到这个集合中;

FD_CLR(int fd,fd_set *fdset);// 清除 fdset 的位 fd,就是将某个位清零;

int FD_ISSET(int fd,fd_set *fdset);// 测试 fdset 的位 fd 是否被设置,其实就是测试某个描述符对应的位是不是1,如果被设置为1返回值为真,否则返回值为假; 

2.3 select第一个参数

select第一个参数通常被设置为描述符的最大值+1

也就是说书上maxfd是描述符的总数目,其实应该理解为描述符的最大值+1;也就是需要关注的位的总数目,比如上面的例子是8;

2.4 select方法之超时时间timeout

timeout 参数用来设置 select 函数的超时时间。它是一个 timeval 结构类型的指针,采用指针参数是因为内核将修改它以告诉应用程序 select 等待了多久。

timeval结构的定义如下:

struct timeval
{
    long tv_sec;//秒数
    long tv_usec;// 微秒数
};

也就是我们能够精确到微秒,实际上它还要看系统能不能达到;

1秒=1000毫秒,1毫秒=1000微秒

如果给 timeout 的两个成员都是 0,则 select 将立即返回。如果 timeout 传递NULL,则 select 将一直阻塞,直到某个文件描述符就绪 

2.5 select方法的用法简述及返回值

我们先将描述符添加到集合中,将集合传参给select,select返回以后我们就要关注它的返回值;

返回值如果等于0,就说明超时了:

如果返回值为n,n>0,那么就说明这个集合中有n(n>0)个描述符有事件就绪;

如果返回值为-1,说明出错了;

2.6 如何检测集合中有哪些描述符有事件就绪

三.select应用

3.1 select小实例

select检查键盘是否有数据

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <assert.h>
#include <unistd.h>
#include <sys/select.h>
#include <sys/time.h>

#define STDIN 0
int main()
{
	int fd=STDIN;
	fd_set fdset;
	while(1)
	{
		FD_ZERO(&fdset);
		FD_SET(fd,&fdset);

		struct timeval tv={5,0};
		int n=select(fd+1,&fdset,NULL,NULL,&tv);
		if(n==-1)
		{
			printf("select error!\n");
			continue;
		}
		else if(n==0)
		{
			printf("time out!\n");
			continue;
		}
		else
		{
			if(FD_ISSET(fd,&fdset))
			{
				char buff[128]={0};
				read(fd,buff,127);
				printf("read:%s\n",buff);
			}
		}
	}
}

 3.2 实现select的tcp服务器端

 #include <stdio.h>
 #include <stdlib.h>
 #include <unistd.h>
 #include <string.h>
 #include <assert.h>
 #include <sys/socket.h>
 #include <arpa/inet.h>
 #include <netinet/in.h>
 #include <fcntl.h>
 #include <sys/select.h>
 #include <sys/time.h>
  
 #define MAXFD 100

int create_socket()
 {
     int sockfd=socket(AF_INET,SOCK_STREAM,0);
     if(sockfd==-1)
     {
         return -1;
     }
 
     struct sockaddr_in saddr;
     memset(&saddr,0,sizeof(saddr));
     saddr.sin_family=AF_INET;
     saddr.sin_port=htons(6000);
     saddr.sin_addr.s_addr=inet_addr("127.0.0.1");
 
     int res=bind(sockfd,(struct sockaddr *)&saddr,sizeof(saddr));
     if(res==-1)
     {
         return -1;
     }
 
     res=listen(sockfd,5);
     if(res==-1)
     {
         return -1;
     }
 
     return sockfd;
 }

 void fds_init(int fds[])
 {
	 for(int i=0;i<MAXFD;i++)
	 {
		 fds[i]=-1;
	 }
 }

 void fds_add(int fds[],int fd)
 {
	 for(int i=0;i<MAXFD;i++)
	 {
		 if(fds[i]==-1)
		 {
			 fds[i]=fd;
			 break;
		 }
	 }
 }

 void fds_del(int fds[],int fd)
 {
	 for(int i=0;i<MAXFD;i++)
	 {
		 if(fds[i]==fd)
		 {
			 fds[i]=-1;
			 break;
		 }
	 }
 }

 int main()
 {
	 int sockfd=create_socket();
	 assert(sockfd!=-1);

	 int fds[MAXFD];
	 fds_init(fds);

	 fds_add(fds,sockfd);

	 fd_set fdset;
	 while(1)
	 {
		 FD_ZERO(&fdset);
		 int maxfd=-1;

		 for(int i=0;i<MAXFD;i++)
		 {
			 if(fds[i]!=-1)
			 {
				 FD_SET(fds[i],&fdset);
				 if(maxfd<fds[i])
				 {
					 maxfd=fds[i];
				 }
			 }
		 }

		 struct timeval tv={5,0};

		 int n=select(maxfd+1,&fdset,NULL,NULL,&tv);
		 if(n==-1)
		 {
			 printf("select error!\n");
			 continue;
		 }
		 else if(n==0)
		 {
			 printf("time out!\n");
			 continue;
		 }
		 else
		 {
			 for(int i=0;i<MAXFD;i++)
			 {
				 if(fds[i]==-1)
				 {
					continue;
				 }
				 if(FD_ISSET(fds[i],&fdset))
				 {
					 if(fds[i]==sockfd)
					 {
						 //accept;
						 struct sockaddr_in caddr;
						 int len=sizeof(caddr);
						 int c=accept(sockfd,(struct sockaddr*)&caddr,&len);
						 if(c==-1)
						 {
							 continue;
						 }
						 printf("accept c=%d\n",c);

						 fds_add(fds,c);
					 }
					 else
					 {
						 //recv
						 char buff[128]={0};
						 int res=recv(fds[i],buff,127,0);
						 if(res<=0)
						 {
							 close(fds[i]);
							 fds_del(fds,fds[i]);
							 printf("one client over!\n");
						 }
						 else
						 {
							 printf("buff(c=%d)=%s\n",fds[i],buff);
							 send(fds[i],"ok",2,0);
						 }
					 }
				 }
			 }
		 }
	 }
	 exit(0);
 }

封装函数版(跟上面的代码作用是一样的) 

//select_ser.c
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <assert.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <sys/select.h>

#define MAX_FD  128
#define DATALEN 1024

//初始化服务器端的sockfd套接字
int InitSocket()
{
    int sockfd=socket(AF_INET,SOCK_STREAM,0);
    if(sockfd==-1)
    {
        return -1;
    }
    
    struct sockaddr_in saddr;
    memset(&saddr,0,sizeof(saddr));
    saddr.sin_family=AF_INET;
    saddr.sin_port=htons(6000);
    saddr.sin_addr.s_addr=inet_addr("127.0.0.1");
    
    int res=bind(sockfd,(struct sockaddr *)&saddr,sizeof(saddr));
    if(res==-1)
    {
        return -1;
    }
    
    res=listen(sockfd,5);
    if(res==-1)
    {
        return -1;
    }
    
    return sockfd;  
}


//初始化记录服务器套接字的数组
void InitFds(int fds[],int n)
{
    int i=0;
    for(;i<n;++i)
    {
        fds[i]=-1;
    }
}


//将套接字描述符添加到数组中
void AddFdToFds(int fds[],int fd,int n)
{
    int i=0;
    for(;i<n;++i)
    {
        if(fds[i]==-1)
        {
            fds[i]=fd;
            break;
        }
    }
}


//删除数组中的套接字描述符
void DelFdFromFds(int fds[],int fd,int n)
{
    int i=0;
    for(;i<n;++i)
    {
        if(fds[i]==fd)
        {
            fds[i]=-1;
            break;
        }
    }
}



//将数组中的套接字描述符设置到fd_set变量中,并返回当前最大的文件描述符值
int SetFdToFdset(fd_set *fdset,int fds[],int n)
{
    FD_ZERO(fdset);
    int i=0,maxfd=fds[0];
    for(;i<n;++i)
    {
        if(fds[i]!=-1)
        {
            FD_SET(fds[i],fdset);
            if(fds[i]>maxfd)
            {
                maxfd=fds[i];
            }
        }
    }
    
    return maxfd;
}

void GetClientLink(int sockfd,int fds[],int n)
{
    struct sockaddr_in caddr;
    memset(&caddr,0,sizeof(caddr));
    socklen_t len=sizeof(caddr);
    int c=accept(sockfd,(struct sockaddr *)&caddr,&len);
    if(c<0)
    {
        return ;
    }
    printf("A client connection was successful\n");
    AddFdToFds(fds,c,n);//新的链接套接字c添加进去集合中
}

//处理客户端数据
void DealClientData(int fds[],int n,int clifd)
{
    char data[DATALEN]={0};
    int num=recv(clifd,data,DATALEN-1,0);
    if(num<=0)
    {
        DelFdFromFds(fds,clifd,n);
        close(clifd);
        printf("A client disconnected\n");
    }
    else
    {
        printf("%d:%s\n",clifd,data);
        send(clifd,"Ok",2,0);
    }
}


//处理select返回的就绪事件
void DealReadyEvent(int fds[],int n,fd_set*fdset,int sockfd)
{
    int i=0;
    for(;i<n;++i)
    {
        if(fds[i]!=-1&& FD_ISSET(fds[i],fdset))
        {
            if(fds[i]==sockfd)
            {
                GetClientLink(sockfd,fds,n);
            }
            else
            {
               DealClientData(fds,n,fds[i]);
            }
        }
    }
}




int main()
{
    int sockfd=InitSocket();
    assert(sockfd!=-1);
    
     fd_set readfds;
     int fds[MAX_FD];
     InitFds(fds,MAX_FD);
     AddFdToFds(fds,sockfd,MAX_FD);
        
     while(1)
     {
         int maxfd=SetFdToFdset(&readfds,fds,MAX_FD);
         struct timeval timeout;
         timeout.tv_sec=5;//秒数
         timeout.tv_usec=0;//微秒
         int n=select(maxfd+1,&readfds,NULL,NULL,&timeout);
         if(n<0)
         {
             printf("select error\n");
             break;
         }
         else if(n==0)
         {
             printf("time  out\n");
             continue;
         }
         
         DealReadyEvent(fds,MAX_FD,&readfds,sockfd);
     }
    exit(0);
    
}

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

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

相关文章

崛起的人工智能,那什么是人工智能?

人工智能是研究、开发用于模拟、延伸和扩展人的智能的理论、方法、技术及应用系统的一门新的技术科学。 人工智能的定义可以分为两部分&#xff0c;即“人工”和“智能”。“人工”比较好理解&#xff0c;争议性也不大。有时我们会要考虑什么是人力所能及制造的&#xff0c;或者…

vscode插件开发-脚手架搭建

安装插件模版 首先我们需要安装Node.js和Git 然后使用以下命令安装Yeoman和VS代码扩展生成器 pnpm install -g yo generator-code 生成脚手架 确保安装好yo和generator-code后&#xff0c;我们使用以下命令生成脚手架 yo code 然后会询问我们想要创建什么类型的扩展、扩…

系列五十五、新版idea Ctrl + Y无法删除代码行问题

一、新版idea Ctrl Y无法删除代码行问题 1.1、idea版本 1.2、解决 https://blog.csdn.net/qq_46921028/article/details/131374801

AI时代Python金融大数据分析实战:ChatGPT让金融大数据分析插上翅膀【文末送书-38】

文章目录 Python驱动的金融智能&#xff1a;数据分析、交易策略与风险管理Python在金融数据分析中的应用 实战案例&#xff1a;基于ChatGPT的金融事件预测AI时代Python金融大数据分析实战&#xff1a;ChatGPT让金融大数据分析插上翅膀【文末送书-38】 Python驱动的金融智能&…

OpenCASCADE开发指南<四>:OCC 数据类型和句柄

一个软件首先要规定能处理的数据类型&#xff0c; 其次要实现三项最基本的功能——引用管理、内存管理和异常管理。在 OCC 中&#xff0c;这三项功能分别对应基础类中的句柄、内存管理器和异常类。 1 数据类型 在基本概念篇里&#xff0c;已经介绍了 OCC 数据类型的分类&…

(2021 AAAI) Self_MM

《Learning Modality-Specific Representations with Self-Supervised Multi-Task Learning for Multimodal Sentiment Analysis》 Abstract 表征学习是多模态学习中一项重要而富有挑战性的任务。有效的模态表示应该包含两部分特征:一致性和差异性。由于多模态标注的统一性,…

Java集合中经典的 5种设计模式,打死也要记住啊!

集合 一、 迭代器模式(Iterator Pattern)二、 工厂模式(Factory Pattern)三、 装饰器模式(Decorator Pattern)四、 适配器模式(Adapter Pattern)五、 组合模式(Composite Pattern) Java 集合框架中的 List、Set、Map 以及其实现类都使用了多种经典的设计模式 一、 迭代器模式(I…

《汇编语言》- 读书笔记 - 第17章-外传之 DOSBox-X 调用 int 13 读写磁盘

《汇编语言》- 读书笔记 - 第17章-外传之 DOSBox-X 调用 int 13 读写磁盘 总结dosbox-x.conf 不完美读取成功写入成功参考资料 总结 DOSBox 中访问 int 13h 始终没反应。网上查了下有人说是没支持&#xff0c;建议使用 DOSBox-X 经过无数遍尝试后&#xff1a; 环境状态Win11…

云计算 3月13号 (OSI 七层模型:物理层、数据链路层、网络层、传输层、会话层、表示层、应用层)

走进网络 1.认识计算机 1.计算机网络是由计算机和通讯构成的&#xff0c;网络研究的是“通信”。 ------1946 世界上第一台计算机 2.终端&#xff1a;只有输入和输出功能&#xff0c;没有计算和处理功能。 3.数据&#xff1a;一串数字&#xff08;二进制数&#xff09;&#…

【el-dialog】解决同一组件使用俩个el-dialog,内容被遮罩层覆盖的问题

如果需要在一个 Dialog 内部嵌套另一个 Dialog或者同一组件有多个Dialog时&#xff0c;需要使用 append-to-body属性 &#xff0c;只要在第二次弹框上面加上属性

第十四届蓝桥杯省赛真题 Java A 组【原卷】

文章目录 发现宝藏【考生须知】试题 A \mathrm{A} A : 特殊日期试题 B: 与或异或试题 C : \mathrm{C}: C: 平均试题 D: 棋盘试题 E : \mathrm{E}: E: 互质数的个数试题 F: 阶乘的和试题 G: 小蓝的旅行计划试题 H: 太阳试题 I: 高塔试题 J \mathrm{J} J : 反异或 01 串 发现…

MongoDB常见面试题总结(一)MongoDB面试题及答案

1. MongoDB的特点&#xff1a; 你能简要介绍一下MongoDB吗&#xff1f;它与关系型数据库的主要区别是什么&#xff1f; MongoDB是一个开源、面向文档的NoSQL数据库&#xff0c;它采用了BSON&#xff08;Binary JSON&#xff09;格式存储数据。以下是MongoDB与关系型数据库的主…

【开发】微服务整合Sentinel

目录 前言 1W&#xff1a;什么是Sentinel&#xff1f; 2W&#xff1a;为什么使用Sentinel&#xff1f; 3W&#xff1a;如何使用Sentinel&#xff1f; 1. 在pom.xml中导入Sentinel依赖坐标 2. 配置控制台 3. 访问API接口的任意端点 流量控制 1. 簇点链路 2. 快速入门…

【HTML】1px边框与1px分割线

对比图 箭头标注的是处理过的 1px分割线 使用transform的scaleY进行缩小 码 <div class"mini-heriz"></div><br><div style"border: solid 1px black; width: 300px;height: 1px;"></div> <style> .mini-heriz {wi…

Java的变量类型详解

目录 局部变量 实例变量 类变量&#xff08;静态变量&#xff09; 参数变量 实例分析 总结 在Java这门静态类型的编程语言中&#xff0c;如何巧妙地使用变量&#xff0c;就像是掌握了一把精准的雕刻刀&#xff0c;能让你在编码的世界里自由地创造。变量在Java中的应用无处…

2024年了,SEO优化是不是已经穷途末路了呢?(川圣SEO)蜘蛛池

baidu搜索&#xff1a;如何联系八爪鱼SEO&#xff1f; baidu搜索&#xff1a;如何联系八爪鱼SEO&#xff1f; baidu搜索&#xff1a;如何联系八爪鱼SEO&#xff1f; 2024年了&#xff0c;SEO优化是不是已经穷途末路了呢&#xff1f;#蜘蛛池SEO SEO优化并没有穷途末路。虽然随…

pcl弧度角度换算:rad2deg,deg2rad

角度弧度换算公式: 代码及结果在:cmath 中cos sin等常用函数的坑(弧度角度换算)-CSDN博客 pcl也有自带的rad2deg,deg2rad: 头文件 #include<pcl/common/angles.h> 代码如下 #include <iostream> #include<pcl/common/angles.h> int main() {vector<…

Linux编程4.3 网络编程-数据封装

1、数据封装 2、Internet协议&#xff08;IP&#xff09; IP的主要目的是为数据输入/输出网络提供基本算法&#xff0c;为高层协议提供无连接的传送服务。这意味着在IP将数据递交给接收站点以前不在传输站点和接收站点之间建立对话&#xff08;虚拟链路&#xff09;。它只是封…

「哈哥赠书活动 - 50期」-『AI赋能写作:AI大模型高效写作一本通』

⭐️ 赠书 - 《AI赋能写作&#xff1a;AI大模型高效写作一本通》 ⭐️ 内容简介 本书以ChatGPT为科技行业带来的颠覆性革新为起点&#xff0c;深入探讨了人工智能大模型如何为我们的创作提供强大支持。本书旨在帮助创作者更好地理解AI的价值&#xff0c;并充分利用其能力提升写…

ubuntu安装开源汇编调试器NASM

安装 安装很简单&#xff0c;直接在终端输入以下命令即可 sudo apt-get install nasm 安装完成后&#xff0c;如果可以查看到nasm的版本号即可视为安装成功 nasm -version 测试 创建汇编文件 创建一个asm文件 vim hello.asm 文件内容如下 section .datahello: db …