I/O复用------select

原创 2016年08月30日 10:57:35

I/O复用使得程序能同时监听多个文件描述符。其适用场合为:

  1. 客户端:同时处理多个socket,同时处理用户输入和为网络链接。
  2. 服务端:同时处理监听、链接socket,同时监听多个端口,同时处理TCP和UDP请求。
    注意:I/O复用本身是阻塞的,当多个描述符同时就绪时,如果不使用多进程/多线程,那么就会按顺序依次处理。

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

1 函数概述

#include <sys/select.h>
int select(int nfds, fd_set* read_fds, fd_set* write_fds, fd_set* exception_fds, struct timeval* timeout);
//nfds 指定被监听文件描述符的总数,通常被设为所监听的最大文件描述符值加1。
//read_fds、write_fds、exception_fds 分别表示可读、可写和异常事件所对应的文件描述符集。
//timeout 设置select函数的超时时间。
//调用成功返回就绪文件描述符总数,失败返回-1。

2 文件描述符的就绪条件
1) 可读:
socket内核接收缓冲区的字节数大于或等于低水位标记SO_RCVLOWAT.
socket通信的对端关闭了链接(此时对socket的读将返回0).
监听socket上有新的链接请求.
socket上有未处理的错误.
2) 可写:
socket内核发送缓冲区的可用字节数大于或等于其低水位标记SO_SNDLOWAT.
socket上的写端被关闭(向被关闭的写端·执行写操作,将触发SIGPIPE信号).
socket使用非阻塞connect连接成功或失败之后.
socket上有未处理的错误.
3)异常
socket上接收到带外数据.

3 处理带外数据

//socket上接收到普通数据和带外数据都将使得socket返回,但是socket会处于不同的就绪状态(可读状态/异常状态)
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <assert.h>
#include <stdio.h>
#include <unistd.h>
#include <errno.h>
#inlcude <string.h>
#include <fcntl.h>
#include <stdlib.h>

#define BUFFER_SIZE 1024

int main(int argc, char *argv[])
{
    if(argc != 3){
        printf("Usage: %s   IP_ADDRESS   PORT_NUMBER\n", basename(argv[0]));
        return 1;
    }
    const char* ip = argv[1];
    int port = atoi(argv[2]);

    struct sockaddr_in address;//填充地址内容
    address.sin_family = AF_INET;
    inet_pton(AF_INET, ip, &address.sin_addr);
    address.sin_port = htons(port);

    int listenfd = socket(PF_INET, SOCK_STREAM, 0);//创建套接字
    assert(listenfd >= 0);

    int ret = bind(listenfd, (struct sockaddr*)&address, sizeof(address));//绑定
    assert(ret != -1);

    //简单例子,服务器只接受一个连接,然后自己进入while循环
    ret = listen(listenfd, 1);
    assert(ret != -1);

    struct sockaddr_in client;
    socklen_t client_length = sizeof(client);
    int connfd = accept(listenfd, (struct sockaddr*)&client, &length_client);//接收
    if(connfd < 0){
        printf("error is: %d\n", errno);
        close(listenfd);
        return 1;
    }
    char buf[BUFFER_SIZE];
    fd_set read_fds;
    fd_set exception_fds;
    FD_ZERO(&read_fds);
    FD_ZERO(&exception_fds);
    while(1){
        memset(buf, '\0', BUFFER_SIZE);
        //内核会动态的修改参数,所以在每次调用select之前,都要重置参数.
        FD_SET(connfd, &read_fds);
        FD_SET(connfd, &exception_fds);
        ret = select(connfd+1, &read_fds, NULL, &exception_fds, NULL);
        if(ret < 0){
            printf("selection failure.\n");
            break;
        }
        if(FD_ISSET(connfd, &read_fds)){
            ret = recv(connfd, buf, BUFFER_SIZE-1, 0);
            if(ret <= 0){
                printf("recv normal data failure.\n");
                break;
            }else{
                printf("get %d bytes of normal data: %s\n", ret, buf);
            }
        }else if(FD_ISSET(connfd, &exception_fds)){
            ret = recv(connfd, buf, BUFFER_SIZE-1, MSG_OOB);
            if(ret <= 0){
                printf("recv oob_data failure.\n");
                break;
            }else{
                printf("get %d bytes oob_data: %s\n", ret, buf);
            }
        }
    }
    close(connfd);
    close(listenfd);
    return 0;   
}
版权声明:本文为博主原创文章,未经博主允许不得转载。

Socket编程实践(8) --Select-I/O复用

五种I/O模型介绍(1)阻塞I/O[默认]   当上层应用App调用recv系统调用时,如果对等方没有发送数据(Linux内核缓冲区中没有数据),上层应用Application1将阻塞;当对等方发送了...
  • hanqing280441589
  • hanqing280441589
  • 2015年03月14日 15:32
  • 3340

Linux网络编程——I/O复用之select详解

一、I/O复用概述 I/O复用概念: 解决进程或线程阻塞到某个 I/O 系统调用而出现的技术,使进程不阻塞于某个特定的 I/O 系统调 I/O复用使用的场合: 1.当客户处理多个描...
  • lianghe_work
  • lianghe_work
  • 2015年06月15日 17:46
  • 2479

Linux网络编程——tcp并发服务器(I/O复用之select

http://blog.csdn.net/lianghe_work/article/details/46519633 与多线程、多进程相比,I/O复用最大的优势是系统开销小,系统不需要建...
  • sinat_35297665
  • sinat_35297665
  • 2017年10月26日 21:02
  • 139

I/O复用select函数

《UNIX网络编程第六章笔记》 1、liunx下可用的5种I/O模型:阻塞式I/O、非阻塞式I/O、I/O复用、信号驱动I/O、异步I/O 1.1、阻塞式I/O 默认情形下,所有套接字都是阻塞的...
  • wustzjf
  • wustzjf
  • 2016年05月14日 11:57
  • 913

I/O复用 select模型使用实例

在这两篇文章基础上 http://blog.csdn.net/qq_26437925/article/details/51898120http://blog.csdn.net/qq_26437925...
  • qq_26437925
  • qq_26437925
  • 2016年08月04日 16:00
  • 289

多路复用I/O模型之select

1 .所谓I/O多路复用是指内核一旦发现进程指定的一个或者多个I/O条件准备读取,它就通知该进程。I/O多路复用适用如下场合:  (1)当客户处理多个描述字时(一般是交互式输入和网络套接口),必须使用...
  • lixiaogang_theanswer
  • lixiaogang_theanswer
  • 2017年06月16日 17:08
  • 200

Linux网络编程——tcp并发服务器(I/O复用之select)

与多线程、多进程相比,I/O复用最大的优势是系统开销小,系统不需要建立新的进程或者线程,也不必维护这些线程和进程。...
  • lianghe_work
  • lianghe_work
  • 2015年06月16日 16:10
  • 1970

select实现I/O多路复用

系统提供select函数来实现多路复用输入/输出模型。select系统调用是用来让我们的程序监视多个文件句柄的状态变化的。 select:该函数允许进程指示内核等待多个事件中的任何一个发生,并只...
  • fucangzxx
  • fucangzxx
  • 2016年07月30日 21:33
  • 1465

linux I/O复用select、poll和epoll

转载自:哈维、dpkirin
  • houliang120
  • houliang120
  • 2014年10月04日 22:13
  • 1652

I/O复用:select、poll、pselect和epoll

一 I/O模型: (1)阻塞式I/O:最常用的I/O模型。默认情况下,所有套接字都是阻塞的。 (2)非阻塞式I/O:不把进程投入睡眠,而是返回一个错误。 (3)I/O复用:调用select或poll,...
  • first_wolf
  • first_wolf
  • 2013年05月25日 21:13
  • 1281
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:I/O复用------select
举报原因:
原因补充:

(最多只允许输入30个字)