SELECT的学习以及在socket中的应用

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* )。一会儿举例说明。 
第二,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秒轮询,要非阻塞就置0 
    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;    //描述符最大值加1 
        switch(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 
文章出处:DIY部落(http://www.diybl.com/course/6_system/linux/Linuxjs/20090308/159832.html) 
linux c语言 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的一个简单用法 

#i nclude <sys/time.h> 
#i nclude <stdio.h> 
#i nclude <sys/types.h> 
#i nclude <sys/stat.h> 
#i nclude <fcntl.h> 
#i nclude <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) 
    { 
timeout.tv_sec=1; 
timeout.tv_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; 
      } 


用来循环读取键盘输入 

2007年9月17日,将例子程序作一修改,加上了time out,并且考虑了select得所有的情况: 

#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) 

      timeout.tv_sec=5; 
      timeout.tv_usec=0; 
      FD_ZERO(&readfd); 
      FD_SET(keyboard,&readfd); 
      ret=select(keyboard+1,&readfd,NULL,NULL,&timeout); 

      //select error when ret = -1 
      if (ret == -1) 
          perror("select error"); 

      //data coming when ret>0 
      else if (ret) 
      { 
          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; 
          } 
      } 

      //time out when ret = 0 
      else if (ret == 0) 
          printf("time out\n"); 




#include <string.h> 
#include <unistd.h> 
#include <sys/time.h> 
#include <sys/types.h> 

下面是我写的一个例程: 
在标准输入读取9个字节数据。 
用select函数实现超时判断! 

int main(int argc, char ** argv) 

char buf[10] = ""; 
fd_set rdfds;// 
struct timeval tv; //store timeout 
int ret; // return val 
FD_ZERO(&rdfds); //clear rdfds 
FD_SET(1, &rdfds); //add stdin handle into rdfds 
tv.tv_sec = 3; 
tv.tv_usec = 500; 
ret = select(1 + 1, &rdfds, NULL, NULL, &tv); 
if(ret < 0) 
perror("\nselect"); 
else if(ret == 0) 
printf("\ntimeout"); 
else 

printf("\nret=%d", ret); 


if(FD_ISSET(1, &rdfds)) 

printf("\nreading"); 
fread(buf, 9, 1, stdin); // read form stdin 

// read(0, buf, 9); /* read from stdin */ 
// fprintf(stdout, "%s\n", buf); /* write to stdout */ 
write(1, buf, strlen(buf)); //write to stdout 
printf("\n%d\n", strlen(buf)); 
return 0; 

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

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

相关文章

python钻石数据分析_数据分析该用什么工具?

数据分析的软件很多&#xff0c;完整的数据分析一般分为数据收集、处理、分析和展现四个步骤。下面分别介绍一下过程中每个步骤使用到的工具。Excel微软办公套装软件的一个重要的组成部分&#xff0c;它包含数据的基本处理&#xff0c;函数计算&#xff0c;数据透视表和VBA等多…

ASP.NET MVC Music Store教程(2):控制器

ASP.NET MVC Music Store教程&#xff08;2)&#xff1a;控制器 转自http://firechun.blog.163.com/blog/static/3180452220110272197830/在传统的Web架构中&#xff0c;URL总是映射到磁盘上的文件。例如&#xff1a;一个类似于“/Products.aspx”或“/Products.php”的URL可能…

C语言:利用泰勒级数计算sinx的值

题目&#xff1a; 代码&#xff1a; #include<stdio.h> #include<math.h> int main(){int sign1,n1;double x3,term,a,sinx0;//scanf("%lf",&x);termx;while(fabs(term)>0.00001){sinxsign*term;nn2;apow(x,n);double b1;for(int i1;i<n;i){…

搞IT也不能不懂“五险一金”啊

养老保险&#xff1a; 一般要交满15年&#xff0c;到退休的时候才能终生享受养老金, 所以想拿养老金的人请务必在自己退休前15年就开始交。如果到退休年龄交养老保险不满15年,那等到你退休的时候国家会把你个人帐户上存的8%的养老金全 部退给你。那单位给你交的21%到哪里去了?…

DecExpress 帮助网站

官方地址&#xff1a;http://devexpress.com 官方下载地址&#xff1a;http://downloads.devexpress.com/8981e647-2ebc-4bd4-b50c-15ca357a62e2/0.0.0.0/DXperience/2011.1/5/DXperienceUniversal-11.1.5.exe 破解地址&#xff1a; smartsoft论坛&#xff1a; http://smartsof…

elementui获取所有树节点_element-ui tree获取子节点全选的父节点信息

公司服务升级确定了新的架构&#xff0c;假如当前部门是二级部门&#xff0c;二级部门下的三级部门全部已选择&#xff0c;那么后端接口要求只需要传二级部门的id&#xff0c;并且操作符传参为 like&#xff1b;如果某一个部门已选择&#xff0c;切父级部门没有选择&#xff0c…

C++课堂整理--第二章内容

提前声明&#xff1a; 本文内容为华北水利水电大学研究生C课程&#xff0c;如有 侵权请告知&#xff0c;作者会予以删除 1程序控制结构 语句是程序的基本语法成分。程序设计语言的语句按功能可以分成三类&#xff1a;声明语句 指示编译器分配内存&#xff0c;或者提供程序…

我的电脑 III

耳机用的是飞利浦的&#xff0c;和上学那时候的差不多 老的那个睡觉的时候压碎了&#xff0c;我胖了 然后去找音箱 网店&#xff0c;专卖店&#xff0c;电脑城&#xff0c;走了一遍又一遍 没个看的上的 一个月以后&#xff0c;终于决定买飞利浦那个了 白色的&#xff0c;和整个…

学习select(函数)

select系统调用是用来让我们的程序监视多个文件句柄(file descrīptor)的状态变化的。程序会停在select这里等待&#xff0c;直到被监视的文件句柄有某一个或多个发生了状态改变。 文件在句柄在代码都是从标准输入读入9个字节字符&#xff1a; #include <stdio.h> #in…

mybatis删除成功返回0_你还在用分页?试试 MyBatis 流式查询,真心强大!

转自&#xff1a;捏造的信仰segmentfault.com/a/1190000022478915基本概念流式查询指的是查询成功后不是返回一个集合而是返回一个迭代器&#xff0c;应用每次从迭代器取一条查询结果。流式查询的好处是能够降低内存使用。如果没有流式查询&#xff0c;我们想要从数据库取 1000…

C++程序设计--第三章内容

提前声明&#xff1a; 本文内容为华北水利水电大学研究生C课程&#xff0c;如有 侵权请告知&#xff0c;作者会予以删除 1.函数 函数作用 —— 任务划分&#xff1b;代码重用定义形式 类型 函数名 &#xff08; 形式参数表&#xff09;{语句序列}调用形式 函数名&#x…

PHP验证变量类型

isset() : //变量是否已经声明empty(): //变量是否为空defined(): //常量是否已经定义 de…

Linux TCP server系列(5)-select模式下的单进程server

目标&#xff1a; 让服务器退化为单进程模式&#xff0c;但是利用select来提升性能 思路&#xff1a; &#xff08;1&#xff09;服务器 传统的单进程服务器一旦accept了客户端的TCP连接后&#xff0c;就转入客户请求的处理&#xff0c;处理完成后才能再一次的调用…

替换元素_80%的前端会答错的问题:lt;imggt;是什么元素?

前言某天晚上&#xff0c;和几个朋友去撸串&#xff0c;突然就聊到了面试&#xff0c;都在感叹现在的面试题太变态了&#xff0c;其中一个突然很神秘的问我&#xff1a;“你写前端这么久了&#xff0c;那你知道 <img> 是什么元素吗&#xff1f;”于是我结合平时写页面的经…

用指针变量访问数组

一维指针 地址值 a 相当于 & a[ 0 ] a 1 相当于 & a[ 1 ] a 2 相当于 & a[ 2 ] a i 相当于 & a[ i ] 元素值 a [ 0 ] * a a [ 1 ] 相当于 * ( a 1 ) a [ 2 ] 相当于 * ( a 2 ) a [ i ] 相当于 * ( a i ) 二维 用指…

LEX和YACC的使用(例子)

1、简单C语言的词法分析程序;%{#include<stdio.h>#include<stdlib.h>#include<ctype.h>#include<string.h> %} digit [0-9]letter [A-Za-z]other_char [!-\[-~]id ({letter}|[_])({letter}|{digit}|[_])*string …

C++排序法

选择排序 // sort void sort ( int x[] , int n ) { int max , t ;for ( int i 0 ; i < n-1 ; i ) //对数组排序{ t i ;for ( int j i 1 ; j < n ; j ) //寻找最大元素if ( x [ j ] > x [ t ] ) t j ;if ( t ! i ){ max x [ i ] ; //交换…

Linux TCP server系列(6)-select模式下的多线程server

目标&#xff1a; 修改上一篇的select模式下的server&#xff0c;让它使用多线程来处理客户端请求&#xff08;多进程的模式已经在上篇中加了注释&#xff09;。 思路&#xff1a; &#xff08;1&#xff09;服务器 我们已经在之前的客户端模型多个并发用户的过程中使用过多线程…

单选按钮_PerlTk教程之按钮Button、复选按钮Checkbutton、单选按钮Radiobutton(附完整代码)...

《Perl-Tk教程之按钮Button、复选按钮Checkbutton、单选按钮Radiobutton》Perl-Tk中有三种不同形式的按钮组件可供选择&#xff0c;它们分别是按钮(Button), 复选按钮(Checkbutton), 和单选按钮(Radiobutton)&#xff0c;如下图所示&#xff1a;这三种按钮看起来是不同的&#…

致读者

有很多理由说明你应该读这个手册&#xff0c;然而也有这样的疑问&#xff1a;为什么要舍近求远&#xff0c;而不是下载一个存在的发行版&#xff1f; 一个很重要的理由是这可以帮助你里里外外的更加了解Linux的工作原理。 另一个关键因素是你可以更多的控制你的系统&#xff0…