linux select函数学习-实现服务器端

 

目录

 

1.select函数的功能和调用顺序

2.设置文件描述符

3.设置监视范围及超时

4.调用select函数后查看结果

5 select函数调用示例

 


1.select函数的功能和调用顺序

使用select函数时可以将多个文件描述符集中到一起统一监视,项目如下:

>是否存在套接字接收数据?

>无需阻塞传输数据的套接字有哪些?

>哪些套接字发生了异常?

2.设置文件描述符

利用select函数可以同时监视多个文件描述符。当然,监视文件描述符可以视为监视套接字。此时首先需要将要监视的文件描述符集中在一起。集中时也要按照监视项(接收 传输 异常)进行区分,即按照上述3种监视项分为3类。

使用fd_set数组变量执行此操作,该数组是存有0或1的位数组。

 在fd_set变量中注册或更改值得操作都有下列宏完成。

FD_ZERO(fd_set * fdset) 将fd_set变量的所有位初始化为0

FD_SET(int fd, fd_set * fdset) 在参数fdset指向的变量中注册文件描述符fd的信息

FD_CLR(int fd, fd_set * fdset) 在参数fdset指向的变量中删除文件描述符fd的信息

FD_ISSET(int fd, fd_set * fdset) 若参数fdset指向的变量中包含文件描述符fd的信息,则返回true

 

3.设置监视范围及超时

 

成功时返回大于0的值,失败时返回-1.

maxfd:监视对象文件描述符数

readset: 将所有关注“是否存在待读取数据”的文件描述符注册到fd_set型变量,并传递其地址值

writeset: 将所有关注“是否可传输无阻塞数据”的文件描述符注册到fd_set型变量,并传递其地址值

readset: 将所有关注“是否发生异常”的文件描述符注册到fd_set型变量,并传递其地址值

timeout:调用select函数后,为防止陷入无限阻塞的状态,传递超时信息

返回值: 发生错误时返回-1,超时返回时返回0.因发生关注的事件返回时,返回大于0的值,该值是发生事件的文件描述符数。

 

在调用select函数前需要决定下面2件事

"文件描述符的监视范围是?"

"如何设定select函数的超时时间"

第一,文件描述符的监视范围与select函数的第一个参数有关。实际上,select函数要求通过第一个函数传递监视对象文件描述符的数量。因此,需要得到注册在fd_set变量中的文件描述符数。但每次新建文件描述符时,其值都会增1,故只需将最大的文件描述符加1再传递到select函数即可。加1时因为文件描述符的值从0开始。

第二,select函数的超时时间与select函数的最后一个参数相关,其中timeval结构体定义如下:

struct timeval
{
    long tv_sec;//seconds
    long tv_usec;//microseconds
}

本来select函数只有在监视的文件描述符发生变化时才返回。如果未发生变化,就会进入阻塞状态。指定超时时间就是为了防止这种情况的发生。通过声明上述结构体变量,将秒数填入tv_sec成员,将微秒数填入tv_usec成员,然后将结构体中的地址值传递到select函数的最后一个参数。此时,即使文件描述符中未发生变化,只要过了指定时间,也可以从函数中返回。不过这种情况下,select函数返回0。因此,可以通过返回值了解返回原因。如果不想设置超时,则传递NULL参数

4.调用select函数后查看结果

select函数返回正整数时,怎样获知那些文件描述符发生了变化?

select函数调用完成后,向其传递的fd_set变量将发生变化。原来为1的所有位均变为0,但发生变化的文件描述符对应位除外。因此,可以认为值仍为1 的位置上的文件描述符发生了变化。

5 select函数调用示例

#include<stdio.h>
#include<unistd.h>
#include<sys/time.h>
#include<sys/select.h>
#define BUF_SIZE 30

int main(int argc, char *argv[])
{
    fd_set reads, temps;
    int result, str_len;
    char buf[BUF_SIZE];
    struct timeval timeout;
    FD_ZERO(&reads);
    FD_SET(0,&reads);//0 is standard input(console)
    
    timeout.tv_sec = 5;
    timeout.tv_usec = 0;
    
    while(1)
    {
        temps = reads;
        printf("tv_sec:%ld,tv_usec:%ld\n",timeout.tv_sec, timeout.tv_usec);
        timeout.tv_sec = 5;
        timeout.tv_usec = 0;
        result = select(1, &temps, 0, 0, &timeout);
        if(result == -1)
        {
            puts("select() error");
            break;
        }
        else if(result == 0)
        {
            puts("time-out");
        }
        else
        {
            if(FD_ISSET(0, &temps))
            {
                str_len = read(0, buf, BUF_SIZE);
                buf[str_len] = 0;
                printf("message from console: %s", buf);
            
            }
        }
    }
    
    return 0;
}
mary@mary:~/code$ ./select
tv_sec:5,tv_usec:0
hello
message from console: hello
tv_sec:1,tv_usec:242701
world
message from console: world
tv_sec:1,tv_usec:617878
time-out
tv_sec:0,tv_usec:0
aaa
message from console: aaa
tv_sec:2,tv_usec:1591
aaaa
message from console: aaaa
tv_sec:2,tv_usec:465709
d
message from console: d
tv_sec:3,tv_usec:313265
time-out
tv_sec:0,tv_usec:0
time-out
tv_sec:0,tv_usec:0

>代码中temps = reads 用于将准备好的fd_set变量reads的内容复制到temps变量,因为调用select函数后,除发生变化的文件描述符对应位外,剩下的所有位将初始化为0。 因此,为了记住初始值,必须经过这种复制过程

>调用select函数后,结构体timeval的成员tv_sec和tv_usec的值将被替换为超时前剩余时间。因此,调用select函数前,每次都需要初始化timeval结构体变量。

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值