linux Select函数用法详解
Linux网络编程学习之select详细讲解

Linux网络编程学习之select()详解select系统调用是用来让我们的程序监视多个文件句柄(file descriptor)的状态变化的。
程序会停在select这里等待,直到被监视的文件句柄有某一个或多个发生了状态改变。
文件在句柄在Linux里很多,如果你man某个函数,在函数返回值部分说到成功后有一个文件句柄被创建的都是的,如man socket可以看到“On success, a file descriptor for the new socket is returned.”而man 2 open可以看到“open() and creat() return the new file descriptor”,其实文件句柄就是一个整数,看socket函数的声明就明白了:int socket(int domain, int type, int protocol);当然,我们最熟悉的句柄是0、1、2三个,0是标准输入,1是标准输出,2是标准错误输出。
0、1、2是整数表示的,对应的FILE *结构的表示就是stdin、stdout、stderr,0就是stdin,1就是stdout,2就是stderr。
比如下面这两段代码都是从标准输入读入9个字节字符:#include <stdio.h>#include <unistd.h>#include <string.h>int main(int argc, char ** argv){char buf[10] = "";read(0, buf, 9); /* 从标准输入0 读入字符*/fprintf(stdout, "%s\n", buf); /* 向标准输出stdout 写字符*/return 0;}/* **上面和下面的代码都可以用来从标准输入读用户输入的9个字符** */#include <stdio.h>#include <unistd.h>#include <string.h>int main(int argc, char ** argv){char buf[10] = "";fread(buf, 9, 1, stdin); /* 从标准输入stdin 读入字符*/write(1, buf, strlen(buf));return 0;}继续上面说的select,就是用来监视某个或某些句柄的状态变化的。
linux 中select的基本用法

linux 中select的基本用法深入了解Linux 中select 的基本用法导语:在Linux 中,select 是一个重要的系统调用,用于同时监视多个文件描述符的可读、可写和出错条件。
它是实现多路复用IO的一种常用方法,能够帮助提高系统的性能。
本文将介绍select 的基本用法,从基础概念到具体使用方法,一步一步讲解,帮助读者更好地理解和应用该系统调用。
第一部分:基础知识1. select 的定义和作用- select 是一个系统调用,用于在一组文件描述符上进行IO 复用- 它能够同时监视多个文件描述符,并在有可读、可写或出错事件发生时进行相应的处理- 使用select 可以减少系统资源的浪费,提高系统的性能2. 文件描述符- 在Linux 中,文件和设备都通过文件描述符来操作- 文件描述符是一个非负整数,用于标识一个打开的文件或设备- 标准输入、标准输出和标准错误输出的文件描述符分别为0、1 和23. fd_set 类型- fd_set 是用于表示文件描述符集合的数据类型- 它通过一个位图来表示文件描述符集合的状态,每个文件描述符占用一个位- 可以使用宏函数来操作fd_set,如FD_ZERO、FD_SET、FD_CLR 和FD_ISSET第二部分:使用步骤1. 初始化文件描述符集合- 使用FD_ZERO 宏函数将文件描述符集合清零- 使用FD_SET 宏函数将需要监视的文件描述符添加到集合中2. 设置超时时间- 声明并初始化timeval 结构体变量,用于指定select 的超时时间- 如果不需要设置超时时间,则将timeval 结构体中的字段都设为03. 调用select 函数- 调用select 函数并传入最大文件描述符数、可读文件描述符集合、可写文件描述符集合、出错文件描述符集合和超时时间- select 函数会阻塞进程,直到有事件发生或超时,返回时会修改文件描述符集合的状态4. 处理事件- 使用FD_ISSET 宏函数判断特定文件描述符是否在集合中- 根据返回的文件描述符集合的状态,进行相应的读、写或出错操作第三部分:注意事项和高级用法1. 最大文件描述符数- select 的第一个参数需要指定最大文件描述符数加1- 如果文件描述符超过了该值,在一些旧版本的Linux 中可能会导致select 函数调用失败2. 文件描述符集合的修改- 在调用select 函数后,文件描述符集合的状态会被修改,只保留了就绪的文件描述符- 因此,每次调用select 函数前都需要重新初始化文件描述符集合3. 非阻塞模式和异步通知- select 在默认情况下是阻塞的,即会一直等待事件发生- 可以通过将文件描述符设置为非阻塞模式来改变这一行为,使得select立即返回- 也可以使用其他方法,如信号处理和管道通信,实现异步通知机制4. poll 和epoll 函数- poll 和epoll 是select 的替代方案,可以更好地处理大量文件描述符- 它们采用事件驱动的方式,不需要每次都重新初始化文件描述符集合- 在需要处理大量并发连接的情况下,可以考虑使用poll 或epoll 函数结语:本文从基础知识到具体使用步骤,详细介绍了Linux 中select 的基本用法。
linux使用select实现精确定时器详解

linux使⽤select实现精确定时器详解在编写程序时,我们经常会⽤到定时器。
⾸先看看select函数原型如下:复制代码代码如下:int select(int nfds, fd_set *readfds, fd_set *writefds,fd_set *exceptfds, struct timeval *timeout);参数说明:slect的第⼀个参数nfds为fdset集合中最⼤描述符值加1,fdset是⼀个位数组,其⼤⼩限制为__FD_SETSIZE(1024),位数组的每⼀位代表其对应的描述符是否需要被检查。
select的第⼆三四个参数表⽰需要关注读、写、错误事件的⽂件描述符位数组,这些参数既是输⼊参数也是输出参数,可能会被内核修改⽤于标⽰哪些描述符上发⽣了关注的事件。
所以每次调⽤select前都需重新初始化fdset。
timeout参数为超时时间,该结构会被内核修改,其值为超时剩余的时间。
利⽤select实现定时器,需要利⽤其timeout参数,注意到:1)select函数使⽤了⼀个结构体timeval作为其参数。
2)select函数会更新timeval的值,timeval保持的值为剩余时间。
如果我们指定了参数timeval的值,⽽将其他参数都置为0或者NULL,那么在时间耗尽后,select函数便返回,基于这⼀点,我们可以利⽤select实现精确定时。
timeval的结构如下:复制代码代码如下:struct timeval{long tv_sec;/*secons*long tv_usec;/*microseconds*/}我们可以看出其精确到microseconds也即微妙。
⼀、秒级定时器复制代码代码如下:void seconds_sleep(unsigned seconds){struct timeval tv;_sec=seconds;_usec=0;int err;do{err=select(0,NULL,NULL,NULL,&tv);}while(err<0 && errno==EINTR);}⼆、毫秒级别定时器复制代码代码如下:void milliseconds_sleep(unsigned long mSec){struct timeval tv;_sec=mSec/1000;_usec=(mSec%1000)*1000;int err;do{err=select(0,NULL,NULL,NULL,&tv);}while(err<0 && errno==EINTR);}三、微妙级别定时器复制代码代码如下:void microseconds_sleep(unsigned long uSec){struct timeval tv;_sec=uSec/1000000;_usec=uSec%1000000;int err;do{err=select(0,NULL,NULL,NULL,&tv);}while(err<0 && errno==EINTR);}现在我们来编写⼏⾏代码看看定时效果吧。
select函数用法

Select在Socket编程中还是比较重要的,可是对于初学Socket的人来说都不太爱用Select写程序,他们只是习惯写诸如connect、accept、recv或recvfrom这样的阻塞程序(所谓阻塞方式block,顾名思义,就是进程或是线程执行到这些函数时必须等待某个事件的发生,如果事件没有发生,进程或线程就被阻塞,函数不能立即返回)。
可是使用Select就可以完成非阻塞(所谓非阻塞方式non-block,就是进程或线程执行此函数时不必非要等待事件的发生,一旦执行肯定返回,以返回值的不同来反映函数的执行情况,如果事件发生则与阻塞方式相同,若事件没有发生则返回一个代码来告知事件未发生,而进程或线程继续执行,所以效率较高)方式工作的程序,它能够监视我们需要监视的文件描述符的变化情况——读写或是异常。
下面详细介绍一下!Select的函数格式(我所说的是Unix系统下的伯克利socket编程,和windows下的有区别,一会儿说明):int select(int maxfdp,fd_set *readfds,fd_set *writefds,fd_set *errorfds,struct timeval *timeout);先说明两个结构体:第一,struct fd_set可以理解为一个集合,这个集合中存放的是文件描述符(filedescriptor),即文件句柄,这可以是我们所说的普通意义的文件,当然Unix下任何设备、管道、FIFO等都是文件形式,全部包括在内,所以毫无疑问一个socket就是一个文件,socket句柄就是一个文件描述符。
fd_set集合可以通过一些宏由人为来操作,比如清空集合FD_ZERO(fd_set *);将一个给定的文件描述符加入集合之中FD_SET(int ,fd_set*);将一个给定的文件描述符从集合中删除FD_CLR(int,fd_set*);检查集合中指定的文件描述符是否可以读写FD_ISSET(int ,fd_set* )。
linux中select、poll、epoll原理

linux中select、poll、epoll原理select、poll和epoll是Linux下常用的I/O多路复用技术,都用于实现高效的事件驱动型的网络编程。
1. select(选择)select是最古老的I/O多路复用机制,它通过在套接字上设置阻塞(阻塞方式)进行等待,一旦有文件描述符准备就绪(可读、可写等),则返回。
select使用fd_set集合来保存要监听的文件描述符,因此其监听的文件描述符数量受到系统给定的FD_SETSIZE限制。
select的实现原理是:在内核中创建一个称为“等待队列”的数据结构(fd_set),该队列保存了需要等待的文件描述符,当某个文件描述符就绪时,会通过和用户进程的映射表通知用户进程。
select通过轮询所有注册的文件描述符,检查哪些文件描述符已经准备好,并将准备好的文件描述符从用户态拷贝到内核态。
select的缺点是每次调用都需要轮询全部的注册文件描述符,效率较低。
2. poll(轮询)poll是在select的基础上进行改进的多路复用技术。
poll与select的最大区别在于,它没有限制文件描述符的数量,并且使用了一个pollfd结构体数组来保存每个文件描述符及其关注的事件。
poll在内核中创建一个称为“等待队列”的数据结构,该队列保存了需要等待的文件描述符,当某个文件描述符就绪时,会通过和用户进程的映射表通知用户进程。
poll的实现原理是:将用户进程注册要监听的文件描述符及其关注的事件存储在内核中的一个事件表中,当发生事件时,内核会将该事件存储在内核态的事件表中,并通知用户进程。
与select不同的是,poll只需在事件发生时拷贝某些信息到内核态,而不需要拷贝全部的文件描述符。
poll的缺点是,当注册的文件描述符数量较大时,每次调用poll都需要遍历整个事件表,效率较低。
3. epoll(事件通知)epoll是Linux特有的一种I/O多路复用机制,通过内核与用户空间的共享内存来实现高效的事件通知。
linux中select、poll、epoll原理详解

linux中select、poll、epoll原理详解目录1. 引言1.1 背景和意义1.2 结构概述1.3 目的2. select原理详解2.1 基本概念2.2 使用方法2.3 应用场景3. poll原理详解3.1 基本概念3.2 使用方法3.3 应用场景4. epoll原理详解4.1 基本概念4.2 使用方法4.3 应用场景5. 结论5.1 对比分析选择合适的IO多路复用器5.2 总结与展望引言1.1 背景和意义在计算机网络编程中,同时监听多个文件描述符的可读、可写和异常事件是一项基本任务。
为了高效地处理这些事件,Linux提供了三种IO多路复用器:select、poll和epoll。
它们允许程序通过一次系统调用就能同时监听多个文件描述符,并在有可读、可写或异常事件发生时进行相应的处理。
使用IO多路复用器可以避免使用阻塞式IO或者轮询方式造成的性能损失,提高了程序的效率和响应速度。
尤其对于具有大量并发连接的服务器程序来说,选择合适的IO多路复用器可以极大地提升系统性能。
1.2 结构概述本文将详细解析Linux中三种IO多路复用器的原理和使用方法,包括select、poll和epoll。
对于每种IO多路复用器,我们将介绍其基本概念、使用方法以及适用场景。
通过深入理解这些IO多路复用器的工作原理,我们可以更好地掌握它们的特点及优缺点,并根据实际需求选择合适的方式来进行网络编程。
1.3 目的本文旨在帮助读者全面了解Linux中select、poll和epoll的原理和使用方法,以及它们在网络编程中的应用场景。
在深入理解这些IO多路复用器的基础上,读者可以根据实际需求灵活选择合适的IO多路复用器,提升程序的性能和可扩展性。
在接下来的文章中,我们将逐一介绍select、poll和epoll的原理详解、使用方法和应用场景,并进行对比分析,最后总结归纳各种IO多路复用器的特点及适用情况。
2. select原理详解2.1 基本概念在Linux系统中,select是一种常用的I/O多路复用机制,它可以监视多个文件描述符的状态是否满足某种条件,在有一或多个文件描述符就绪时通知进程进行相应的 I/O操作。
Linux中select函数学习及实例笔记

Linux中select函数学习及实例笔记Unix中的函数select和poll用来,支持Unix中I/O复用的功能,在Unix中I/O模型可以分为以一几种:(1)阻塞I/O(2)非阻塞I/O(3)I/O复用(select和poll)(4)信号驱动I/O(SIGIO)(5)异步I/O其中,现在比较流行的I/O模型是阻塞I/O模型.阻塞I/O是当应用程序和内核交换数据时,由于内核还没有准备好数据,那么应用程序必须进行阻塞,不能继续执行,直到内核的数据准备好!应用程序取到数据返回后,阻塞过程结束!但返回的结果也并不一定是正确的!这里只是举一个简单的例子!也许情况会更加的复杂!非阻塞I/O,例如在和内核交换数据时,如果内核的数据没有准备好,那么应用程序不会一真等待,会有一个返回信息,以判断是那里出了问题!这样有助于确认是在那个阶段出了问题!I/O复用,我们就可以调用系统调用select和poll!在这两个系统调用中的某一个阻塞,而不是真正的阻塞I/O系统调用!下面主要介绍I/O复用中的select函数!select函数可以指示内核等待多个事件中的任一个发生,仅在一个或多个事件发生,或者等待一个足够的时间后才唤醒进程!select函数的原型如下:#include <sys/types.h>#include<sys/time.h>int select (int maxfdp1,fd_set *readset,fd_set * writeset,fd_set excpetset,const struct timeval *timeout);返回值:准备好的描述符的正数目 0---超时 -1---出错!其中最后一个参数是一个结构体的指针,它表示等待内核中的一组描述符任一个准备好需要花费多久的时间!其中timeval指定了秒数和微秒数。
struct timeval{long tv_sec;//秒数long tv_usec;//微秒数};将 timeout设置为空指针时,会永远等待下去,等待固定的时间:如果timeout指向的timeval中的具体的值时,会等待一个固定的时间,不等待立刻返回,这时timeval中的tv_sec和tv_usec为0.select有三个可能的返回值。
select语句的基本用法总结

select语句的基本用法总结
嘿,朋友!你知道吗,在数据库的世界里,select 语句就像是一把神奇的钥匙,能帮我们打开数据宝库的大门!
比如说,你想要从一个包含众多学生信息的表格里,找出所有成绩在 90 分以上的同学,这时候 select 语句就派上用场啦!它能像一个超级侦探一样,精准地把你想要的信息给揪出来。
select 语句的基本语法就像是搭积木一样,简单又有趣。
它通常是这样的:“SELECT 列名 FROM 表名” 。
这就好比你去超市买东西,先告诉售货员你想要啥(列名),然后再说从哪个货架上拿(表名)。
你想想,如果不告诉数据库你想要哪些列的数据,那不就像在大海里捞针,完全摸不着头脑嘛?
再比如说,你只想获取特定条件下的数据,比如只想要年龄大于 20 岁的人的信息,那就可以在后面加上“WHERE 条件”。
这就好像是给你的搜索加上了一个精准的过滤器,把不符合条件的统统筛掉!
而且啊,select 语句还能进行排序呢!用“ORDER BY 列名”就能让数据按照你指定的顺序排列,这不就跟整理书架一样,想让书按照作者名字排或者出版年份排,随你心意!
怎么样,是不是觉得 select 语句超级厉害?
在我看来呀,掌握好 select 语句,就等于掌握了在数据库世界里畅游的重要技能,能让我们轻松获取所需的数据,解决各种问题!。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
Socket-SelectSelect在Socket编程中还是比较重要的,可是对于初学Socket的人来说都不太爱用Select写程序,他们只是习惯写诸如 connect、accept、recv或recvfrom 这样的阻塞程序(所谓阻塞方式block,顾名思义,就是进程或是线程执行到这些函数时必须等待某个事件的发生,如果事件没有发生,进程或线程就被阻塞,函数不能立即返回)。
可是使用Select就可以完成非阻塞(所谓非阻塞方式non-block,就是进程或线程执行此函数时不必非要等待事件的发生,一旦执行肯定返回,以返回值的不同来反映函数的执行情况,如果事件发生则与阻塞方式相同,若事件没有发生则返回一个代码来告知事件未发生,而进程或线程继续执行,所以效率较高)方式工作的程序,它能够监视我们需要监视的文件描述符的变化情况——读写或是异常。
下面详细介绍一下。
Select的函数格式:int select(int maxfdp,fd_set *readfds,fd_set *writefds,fd_set*errorfds,struct timeval *timeout);先说明两个结构体:第一,struct fd_set可以理解为一个集合,这个集合中存放的是文件描述符(file descriptor),即文件句柄,这可以是我们所说的普通意义的文件,当然Unix下任何设备、管道、FIFO等都是文件形式,全部包括在内,所以毫无疑问一个socket就是一个文件,socket句柄就是一个文件描述符。
fd_set集合可以通过一些宏由人为来操作,比如清空集合 FD_ZERO(fd_set *),将一个给定的文件描述符加入集合之中FD_SET(int ,fd_set *),将一个给定的文件描述符从集合中删除FD_CLR(int ,fd_set*),检查集合中指定的文件描述符是否可以读写FD_ISSET(int ,fd_set* )。
一会儿举例说明。
第二,struct timeval是一个大家常用的结构,用来代表时间值,有两个成员,一个是秒数,另一个是毫秒数。
具体解释select的参数:int maxfdp是一个整数值,是指集合中所有文件描述符的范围,即所有文件描述符的最大值加1,不能错!在Windows中这个参数的值无所谓,可以设置不正确。
fd_set *readfds是指向fd_set结构的指针,这个集合中应该包括文件描述符,我们是要监视这些文件描述符的读变化的,即我们关心是否可以从这些文件中读取数据了,如果这个集合中有一个文件可读,select就会返回一个大于0的值,表示有文件可读,如果没有可读的文件,则根据timeout参数再判断是否超时,若超出timeout的时间,select返回0,若发生错误返回负值。
可以传入NULL值,表示不关心任何文件的读变化。
fd_set *writefds是指向fd_set结构的指针,这个集合中应该包括文件描述符,我们是要监视这些文件描述符的写变化的,即我们关心是否可以向这些文件中写入数据了,如果这个集合中有一个文件可写,select就会返回一个大于0的值,表示有文件可写,如果没有可写的文件,则根据timeout参数再判断是否超时,若超出timeout的时间,select返回0,若发生错误返回负值。
可以传入NULL值,表示不关心任何文件的写变化。
fd_set *errorfds同上面两个参数的意图,用来监视文件错误异常。
struct timeval* timeout是select的超时时间,这个参数至关重要,它可以使select处于三种状态,第一,若将NULL以形参传入,即不传入时间结构,就是将select置于阻塞状态,一定等到监视文件描述符集合中某个文件描述符发生变化为止;第二,若将时间值设为0秒0毫秒,就变成一个纯粹的非阻塞函数,不管文件描述符是否有变化,都立刻返回继续执行,文件无变化返回0,有变化返回一个正值;第三,timeout的值大于0,这就是等待的超时时间,即 select 在timeout时间内阻塞,超时时间之内有事件到来就返回了,否则在超时后不管怎样一定返回,返回值同上述。
返回值:负值:select错误正值:某些文件可读写或出错0:等待超时,没有可读写或错误的文件在有了select后可以写出像样的网络程序来!举个简单的例子,就是从网络上接受数据写入一个文件中。
例子:main(){int sock;FILE *fp;struct fd_set fds;struct timeval timeout={3,0}; //select等待3秒,3秒轮询,要非阻塞就置char buffer[256]={0}; //256字节的接收缓冲区/* 假定已经建立UDP连接,具体过程不写,简单,当然TCP也同理,主机ip 和port都已经给定,要写的文件已经打开sock=socket(...);bind(...);fp=fopen(...); */while(1){FD_ZERO(&fds); //每次循环都要清空集合,否则不能检测描述符变化FD_SET(sock,&fds); //添加描述符FD_SET(fp,&fds); //同上maxfdp=sock>fp?sock+1:fp+1; //描述符最大值加1switch(select(maxfdp,&fds,&fds,NULL,&timeout)) //select使用{case -1: exit(-1);break; //select错误,退出程序case 0:break; //再次轮询default:if(FD_ISSET(sock,&fds)) //测试sock是否可读,即是否网络上有数据{recvfrom(sock,buffer,256,.....);//接受网络数据if(FD_ISSET(fp,&fds)) //测试文件是否可写fwrite(fp,buffer...);//写入文件//buffer清空;} // end if break;} // end switch} //end while} //end main由于采用select 机制, 因此当没有字符可读时, 程序处于阻塞状态,最小程度的占用CPU 资源, 在同一台机器上执行一个server 和若干个client 时, 系统负载只有0.1 左右, 而采用原来的非阻塞通信方法, 只运行一个server, 系统负载就可以达到1.5 左右. 因此我们推荐使用select。
socket select()用法头文件#i nclude<sys/time.h>#i nclude<sys/types.h>#i nclude<unistd.h>定义函数int select(int n,fd_set * readfds,fd_set * writefds,fd_set * exceptfds,struct timeval * timeout);函数说明select()用来等待文件描述词状态的改变。
参数n代表最大的文件描述词加1,参数readfds、writefds 和exceptfds 称为描述词组,是用来回传该描述词的读,写或例外的状况。
底下的宏提供了处理这三种描述词组的方式:FD_CLR(inr fd,fd_set* set);用来清除描述词组set中相关fd 的位FD_ISSET(int fd,fd_set *set);用来测试描述词组set中相关fd 的位是否为真FD_SET(int fd,fd_set*set);用来设置描述词组set中相关fd的位FD_ZERO(fd_set *set);用来清除描述词组set的全部位参数timeout为结构timeval,用来设置select()的等待时间,其结构定义如下struct timeval{time_t tv_sec;time_t tv_usec;};返回值如果参数timeout设为NULL则表示select()没有timeout。
错误代码执行成功则返回文件描述词状态已改变的个数,如果返回0代表在描述词状态改变前已超过timeout时间,当有错误发生时则返回-1,错误原因存于errno,此时参数readfds,writefds,exceptfds和timeout的值变成不可预测。
EBADF 文件描述词为无效的或该文件已关闭EINTR 此调用被信号所中断EINVAL 参数n 为负值。
ENOMEM 核心内存不足范例常见的程序片段:fs_set readset;FD_ZERO(&readset);FD_SET(fd,&readset);select(fd+1,&readset,NULL,NULL,NULL);if(FD_ISSET(fd,readset){……}下面是linux环境下select的一个简单用法#include <sys/time.h>#include <stdio.h>#include <sys/types.h>#include <sys/stat.h>#include <fcntl.h>#include <assert.h>int main (){int keyboard;int ret,i;char c;fd_set readfd;struct timeval timeout;keyboard = open("/dev/tty",O_RDONLY | O_NONBLOCK);assert(keyboard>0);while(1){_sec=1;_usec=0;FD_ZERO(&readfd);FD_SET(keyboard,&readfd);ret=select(keyboard+1,&readfd,NULL,NULL,&timeout); if(FD_ISSET(keyboard,&readfd)){i=read(keyboard,&c,1);if('\n'==c)continue;printf("hehethe input is %c\n",c);if ('q'==c)break;}}}。