IO多路复用
一、概念
IO多路复用技术 是一种 网络通信 的方式,通过这种方式可以同时检测多个 文件描述符(这个过程是阻塞的),一旦检测到某一个文件描述符(状态是 可读 或者 可写 的)是就绪的,那么就会解除阻塞,然后我们就可以基于这些已经就绪的文件描述符进行网络通信了。
通过这种方式,服务端即使是在 单线程/进程 的情况下也能实现并发,支持多个连接。
常用的 IO多路复用 的方式有 : select 、 poll 、epoll。
二、IO多路复用 与 传统的多线程/进程方式 对比
传统的 多线程/进程 并发
主线程 / 父进程 调用 accept() 监测是否有客户端的连接到来:
- 如果没有客户端连接请求到来,那么当前 线程/进程 就会阻塞在这里;
- 如果有客户端连接请求到来,那么就先解除阻塞,建立新连接;
子线程 / 子进程 调用 send() / write() , read() / recv() 和客户端建立的连接通信 :
- 调用 read() / recv()接收客户端发送过来的通信数据,如果 读缓冲区 中没有数据,那么当前 线程/进程 会阻塞,直到 读缓冲区 中有了数据,阻塞就会自动解除;
- 调用 send() / write()向客户端发送数据,如果 写缓冲区 的容量已经满了,那么当前 线程/进程 会阻塞,直到 写缓冲区 有了空间,阻塞就会自动解除;
IO多路复用并发
使用 IO多路复用函数 会 委托内核 检测所有客户端的文件描述符 (主要是用于 监听 和 通信 的两类),这个过程会导致 进程/线程 的阻塞;如果内核检测到有就绪的文件描述符就会解除阻塞,然后将这些已经就绪的文件描述符传出去。
然后根据被传出的文件描述符的类型不同,做不同的处理:
1.用于监听的文件描述符 lfd :用于和客户端建立连接
- 此时再调用 accept()和客户端建立新连接是不会阻塞的,因为此时用于监听的文件描述符lfd是就绪的,也就是它对应的 读缓冲区 中有连接请求;
2.用于通信的文件描述符 cfd :调用通信函数 和 已经建立连接的客户端进行通信
- 调用 read() / recv()不会阻塞,因为此时用于通信的文件描述符cfd是就绪的,它对应的 读缓冲区 中有数据;
- 调用 send() / write()不会阻塞,因为此时用于通信的文件描述符cfd是就绪的,它对应的 写缓冲区 中有多余的容量;
3.对这些文件描述符继续进行下一轮检测,一直循环下去…
与 多线程/进程 技术相比,IO多路复用的优势在与不用频繁的进行 线程/进程的创建和销毁,不用管理 线程/进程,极大的减少了资源的消耗。
三、三种IO多路复用的方式
1.select
select 是跨平台的,同时支持 Linux 、 Windows 、MacOS 。我们通过调用 select() 这个函数就可以委托内核帮助我们检测 文件描述符 的状态,也就是检测这些文件描述符对应的 读写缓冲区 的状态:
- 读缓冲区: 检测里面有没有数据,如果有的话,说明其对应的文件描述符已经就绪了;
- 写缓冲区: 检测 写缓冲区 有没有多余的容量可以写,如果有就说明这个 写缓冲区 对应的文件描述符已经就绪了;
- 读写异常: 检测 读写缓冲区 是否有异常,如果有的话说明该缓冲区对应的文件描述符已经就绪了;
内核检测完毕文件描述符的状态之后,已经就绪的文件描述符会通过 select() 的三个参数传出,这三个参数都是一个 集合,我们得到之后就能对其进行处理。
1.函数原型
#include <sys/select.h>
struct timeval {
    time_t      tv_sec;         /* seconds */
    suseconds_t tv_usec;        /* microseconds */
};
int select(int nfds, fd_set *readfds, fd_set *writefds,
           fd_set *exceptfds, struct timeval * timeout);
nfds : 委托内核检测的这三个集合中 最大的文件描述符 + 1
- 内核需要遍历集合这三个集合中的文件描述符,这个值就是循环结束的条件;
- 在 Windows中这个参数是无效的,直接指定为 − 1 -1 −1 即可;
readfds:文件描述符的集合,内核只检测这个集合中的文件描述符对应的 读缓冲区
- 这是一个传入传出参数,读集合一般情况下都是需要检测的,这样才知道是通过那个文件描述符接收数据;
wtitedfs:文件描述符的集合,内核只检测这个集合中的文件描述符对应的 写缓冲区
- 这是一个传入传出参数,如果不需要这个参数可以指定为 NULL;
exceptdfs:文件描述符的集合,内核只检测这个集合中的文件描述符是否有异常状态
- 这是一个传入传出参数,如果不需要这个参数可以指定为 NULL;
timeout :超时时长,用来强制解除 select() 的阻塞的
- 如果指定为 NULL的话,select()检测不到就绪的文件描述符就会一直阻塞;
- 等待固定时长:select()检测不到就绪的文件描述符,在指定时长之后强制解除阻塞,返回 0 0 0;
- 不等待:直接将该参数的结构体指定为  
      
       
        
        
          0 
         
        
       
         0 
        
       
     0 ,select()函数就不会阻塞;
返回值:
- 大于 0 0 0 ,成功。直接返回集合中已经就绪的文件描述符的总个数;
- 等于 − 1 -1 −1,失败;
- 等于 0 0 0,超时。没有检测到就绪的文件描述符;
需要使用到的一些函数:
// 将文件描述符fd从set集合中删除 == 将fd对应的标志位设置为0        
void FD_CLR(int fd, fd_set *set);
// 判断文件描述符fd是否在set集合中 == 读一下fd对应的标志位到底是0还是1
int  FD_ISSET(int fd, fd_set *set);
// 将文件描述符fd添加到set集合中 == 将fd对应的标志位设置为1
void FD_SET(int fd, fd_set *set);
// 将set集合中, 所有文件文件描述符对应的标志位设置为0, 集合中没有添加任何文件描述符
void FD_ZERO(fd_set *set);
2.实现细节
在 select() 函数中第2、3、4个参数都是 fd_set 类型,它表示一个文件描述符的集合,这个类型的数据有  
     
      
       
       
         128 
        
       
      
        128 
       
      
    128 个字节,也就是  
     
      
       
       
         1024 
        
       
      
        1024 
       
      
    1024 个标志位,和内核中文件描述符表中的文件描述符个数是一样的。
sizeof(fd_set) = 128 字节 * 8 = 1024 bit      // int [32]
这块内存中的每一个bit 和 文件描述符表中的每一个文件描述符是一一对应的关系,这样就可以使用最小的存储空间将要表达的意思描述出来了。
下图中的 fd_set 存储了要委托内核要检测的 读缓冲区的文件描述符集合。
- 如果集合中的 标志位为 0 0 0 ,代表 不检测 这个文件描述符的状态;
- 如果集合中的 标志位为 1 1 1 ,代表 检测 这个文件描述符的状态;

 内核在遍历这个 读集合 的过程中,如果被检测的文件描述符对应的读缓冲区中 没有数据,内核将修改这个文件描述符在读集合 fd_set 中对应的标志位,改为  
     
      
       
       
         0 
        
       
      
        0 
       
      
    0,如果有数据那么这个标志位的值不变,还是 
     
      
       
       
         1 
        
       
      
        1 
       
      
    1。

 当 select() 函数解除阻塞之后,被内核修改过的 读集合 通过参数传出,此时集合中只要标志位的值为  
     
      
       
       
         1 
        
       
      
        1 
       
      
    1,那么它对应的文件描述符肯定是就绪的,我们就可以基于这个文件描述符和客户端建立新连接或者通信了。
3.处理流程
1.创建监听的套接字 lfd = socket() ;
 2.将 监听的套接字 和 本地的 ip 和 端口 绑定 bind();
 3.给监听的套接字设置监听 listen();
 4.创建一个文件描述符集合 fd_set,用于存储需要检测读事件的所有的文件描述符
 通过 FD_ZERO() 初始化;
- 通过 FD_SET()将监听的文件描述符放入检测的读集合中
- 循环调用 select(),周期性的对所有的文件描述符进行检测
5.select() 解除阻塞返回,得到内核传出的满足条件的就绪的文件描述符集合
 6.通过 FD_ISSET() 判断集合中的标志位是否为  
     
      
       
       
         1 
        
       
      
        1 
       
      
    1
- 如果这个文件描述符是 监听的文件描述符,调用 accept()和客户端建立连接。将得到的新的通信的文件描述符,通过FD_SET()放入到检测集合中
- 如果这个文件描述符是通信的文件描述符,调用通信函数和客户端通信
-  
  - 如果客户端和服务器断开了连接,使用 FD_CLR()将这个文件描述符从检测集合中删除
 
- 如果客户端和服务器断开了连接,使用 
-  
  - 如果没有断开连接,正常通信即可
 
7.重复第6步

4.实现
客户端代码:
#include <stdio.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <string.h>
int main(){
    //1.创建用于通信的文件描述符 cfd
    int cfd = socket(AF_INET,SOCK_STREAM,0);
    if(cfd == -1){
        perror("socket");
        return -1;
    }
    printf("1.成功创建了用于通信的文件描述符 : %d\n",cfd);
    //2.连接服务器
    unsigned short port = 10000;
    
    //你自己的服务器 或者 虚拟机的 ip地址
    const char* ip = "10.0.8.14";
    struct sockaddr_in addr;
    addr.sin_family = AF_INET;
    addr.sin_port = htons(port);
    inet_pton(AF_INET,ip,&addr.sin_addr.s_addr);
    int ret = connect(cfd,(struct sockaddr*)&addr,sizeof(addr));
    if(ret == -1){
        perror("connet");
        return -1;
    }
    printf("2.成功连接了服务器 , ip : %s , port : %d\n",ip,port);
    //3.开始通信
    char send_buf[1024];
    char recv_buf[1024];
    int cnt = 0;
    while(1){
        memset(send_buf,0,sizeof send_buf);
        memset(recv_buf,0,sizeof recv_buf);
        sprintf(send_buf,"hello i love you : %d",cnt++);
        //发送数据
        send(cfd,send_buf,strlen(send_buf) + 1,0);
        //接收数据
        int len = recv(cfd,recv_buf,sizeof(recv_buf),0);
        if(len > 0){
            printf("服务端 : %s\n",recv_buf);
        }
        else if(len == 0){
            printf("服务端已经断开了连接...\n");
            break;
        }
        else{
            perror("recv");
            break;
        }
        sleep(1);
    }
    close(cfd);
    return 0;
}
1.服务端基础版本
#include <stdio.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <pthread.h>
#include <strings.h>
#include <string.h>
#include <sys/select.h>
int main()
{
    // 1.创建用于监听的文件描述符
    int lfd = socket(AF_INET, SOCK_STREAM, 0);
    if (lfd == -1)
    {
        perror("socket");
        return -1;
    }
    // 2.绑定 ip 和 端口号
    unsigned short port = 10000;
    struct sockaddr_in saddr;
    saddr.sin_family = AF_INET;
    saddr.sin_port = htons(port);
    saddr.sin_addr.s_addr = INADDR_ANY;
    int ret = bind(lfd, (struct sockaddr *)&saddr, sizeof(saddr));
    if (ret == -1)
    {
        perror("bind");
        return -1;
    }
    // 3.设置监听
    ret = listen(lfd, 128);
    if (ret == -1)
    {
        perror("listen");
        return -1;
    }
    printf("设置监听成功...\n");
    // 4.获取连接
    fd_set readset ,temp;
    FD_ZERO(&readset);
    //把用于监听的文件描述符 lfd , 加入到 readset 读集合中
    FD_SET(lfd,&readset);
    int addr_len = sizeof(struct sockaddr_in);
    int maxfd = lfd;
    
    char buf[1024];
    char* str = "ok";
    while(1){
         temp = readset;
        int ret = select(maxfd + 1,&temp,NULL,NULL,NULL);
        //检测 用于监听的文件描述符 lfd 读缓冲区是否有数据 , 也就是是否有新的连接到来
        if(FD_ISSET(lfd,&temp)){
            struct sockaddr_in addr;
            int cfd = accept(lfd,(struct sockaddr*)&addr,&addr_len);
            //将用于通信文件描述符 cfd 也加入到 读集合中
            FD_SET(cfd,&readset);
            //更新 maxfd
            maxfd = cfd > maxfd ? cfd : maxfd;
            printf("获取连接成功 , 客户端 ip : %s  , port : %d\n", inet_ntoa(addr.sin_addr), ntohs(addr.sin_port));
        }
        for(int i = 0;i <= maxfd;i++){
            //用于通信的文件描述符 cfd 的读缓冲区有数据了 , 也就是有客户端发消息过来了
            if(i != lfd && FD_ISSET(i,&temp)){
                
                memset(buf,0,sizeof buf);
                int len = read(i,buf,sizeof buf);
                printf("客户端 : %s\n",buf);
                if(len > 0){
                    write(i,str,strlen(str) + 1);
                }
                else if(len == 0){
                    //客户端已经关闭了连接
                    printf("客户端已经关闭了连接...\n");
                    FD_CLR(i,&readset);
                    close(i);
                }
                else{
                    perror("read");
                }
            }
        }
    }
    close(lfd);
    return 0;
}
2.服务端多线程版本
#include <stdio.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <pthread.h>
#include <strings.h>
#include <string.h>
#include <stdlib.h>
#include <sys/select.h>
#include <pthread.h>
#include <ctype.h>
pthread_mutex_t mutex;
char buf[1024];
typedef struct fdinfo
{
    int fd;
    int *maxfd;
    fd_set *readset;
} FdInfo;
void *acceptConnection(void *arg)
{
    printf("子线程的线程id为 : %ld\n", pthread_self());
    FdInfo *info = (FdInfo *)(arg);
    int lfd = info->fd;
    struct sockaddr_in addr;
    int addr_len = sizeof(addr);
    int cfd = accept(lfd, (struct sockaddr *)&addr, &addr_len);
    pthread_mutex_lock(&mutex);
    // 将用于通信文件描述符 cfd 也加入到 读集合中
    FD_SET(cfd, info->readset);
    // 更新 maxfd
    *info->maxfd = cfd > *info->maxfd ? cfd : *info->maxfd;
    pthread_mutex_unlock(&mutex);
    printf("获取连接成功 , 客户端 ip : %s  , port : %d\n", inet_ntoa(addr.sin_addr), ntohs(addr.sin_port));
    free(info);
    return NULL;
}
void *communicate(void *arg)
{
    printf("子线程的线程id为 : %ld\n", pthread_self());
    FdInfo *info = (FdInfo *)(arg);
    int cfd = info->fd;
    memset(buf, 0, sizeof buf);
    int len = read(cfd, buf, sizeof buf);
    printf("客户端 : %s\n", buf);
    if (len < 0)
    {
        perror("read");
        free(info);
        return NULL;
    }
    else if(len == 0){
        
        // 客户端已经关闭了连接
        printf("客户端已经关闭了连接...\n");
        pthread_mutex_lock(&mutex);
        FD_CLR(cfd, info->readset);
        pthread_mutex_unlock(&mutex);
        close(cfd);
        free(info);
        
        return NULL;
    }
    int str_len = strlen(buf);
    for(int i = 0;i < str_len;i++) buf[i] = toupper(buf[i]);
    write(cfd,buf,len);
    free(info);
    return NULL;
}
int main()
{
    // 初始化 mutex
    pthread_mutex_init(&mutex, NULL);
    // 1.创建用于监听的文件描述符
    int lfd = socket(AF_INET, SOCK_STREAM, 0);
    if (lfd == -1)
    {
        perror("socket");
        return -1;
    }
    // 2.绑定 ip 和 端口号
    unsigned short port = 10000;
    struct sockaddr_in saddr;
    saddr.sin_family = AF_INET;
    saddr.sin_port = htons(port);
    saddr.sin_addr.s_addr = INADDR_ANY;
    int ret = bind(lfd, (struct sockaddr *)&saddr, sizeof(saddr));
    if (ret == -1)
    {
        perror("bind");
        return -1;
    }
    // 3.设置监听
    ret = listen(lfd, 128);
    if (ret == -1)
    {
        perror("listen");
        return -1;
    }
    printf("设置监听成功...\n");
    // 4.获取连接
    fd_set readset, temp;
    FD_ZERO(&readset);
    // 把用于监听的文件描述符 lfd , 加入到 readset 读集合中
    FD_SET(lfd, &readset);
    int addr_len = sizeof(struct sockaddr_in);
    int maxfd = lfd;
    while (1)
    {
        pthread_mutex_lock(&mutex);
        temp = readset;
        pthread_mutex_unlock(&mutex);
        int ret = select(maxfd + 1, &temp, NULL, NULL, NULL);
        // 检测 用于监听的文件描述符 lfd 读缓冲区是否有数据 , 也就是是否有新的连接到来
        if (FD_ISSET(lfd, &temp))
        {
            FdInfo *info = (FdInfo *)malloc(sizeof(FdInfo));
            info->fd = lfd;
            info->maxfd = &maxfd;
            info->readset = &readset;
            pthread_t tid;
            pthread_create(&tid, NULL, acceptConnection, info);
            pthread_detach(tid);
        }
        for (int i = 0; i <= maxfd; i++)
        {
            // 用于通信的文件描述符 cfd 的读缓冲区有数据了 , 也就是有客户端发消息过来了
            if (i != lfd && FD_ISSET(i, &temp))
            {
                FdInfo *info = (FdInfo *)malloc(sizeof(FdInfo));
                info->fd = i;
                info->maxfd = &maxfd;
                info->readset = &readset;
                pthread_t tid;
                pthread_create(&tid, NULL, communicate, info);
                pthread_detach(tid);
            }
        }
    }
    close(lfd);
    pthread_mutex_destroy(&mutex);
    return 0;
}
客户端不需要使用IO多路复用进行处理,因为客户端和服务器的对应关系是 1 : N 1:N 1:N,也就是说客户端是比较专一的,只能和一个连接成功的服务器通信。
虽然使用select这种IO多路转接技术可以降低系统开销,提高程序效率,但是它也有局限性:
- 待检测集合(第 2 、 3 、 4 2、3、4 2、3、4 个参数)需要频繁的在用户区和内核区之间进行数据的拷贝,效率低
- 内核对于 select()传递进来的待检测集合的检测方式是线性的
-  
  - 如果集合内待检测的文件描述符很多,检测效率会比较低
 
-  
  - 如果集合内待检测的文件描述符相对较少,检测效率会比较高
 
- 使用 select能够检测的最大文件描述符个数有上限,默认是 1024 1024 1024,这是在内核中被写死了的。
2.poll
poll 的机制跟 select 类似,使用方法也是类似的,以下是它们两个的对比:
- 内核检测文件描述符的状态也是通过 线性遍历 的形式;
- poll和- select检测的文件描述符的集合 会在被检测的过程频繁的进行 用户区 和 内核区的拷贝,它的开销随着文件描述符数量的增加而增大,所以效率也会变得越来越低;
- select可以跨平台使用,支持- Linux、- Windows、- MacOS;而- poll只能在- Linux平台下使用;
1.函数原型
#include <poll.h>
// 每个委托poll检测的fd都对应这样一个结构体
struct pollfd {
    int   fd;         /* 委托内核检测的文件描述符 */
    short events;     /* 委托内核检测文件描述符的什么事件 */
    short revents;    /* 文件描述符实际发生的事件 -> 传出 */
};
struct pollfd myfd[100];
int poll(struct pollfd *fds, nfds_t nfds, int timeout);
fds :这是一个 struct pollfd 类型的数组,里面存储了待检测的文件描述符的信息,它一共有三个成员:
- fd: 委托内核检测的文件描述符;
- events:委托内核检测的文件描述符对应的事件(读,写,错误);
- revents:这是一个传出参数,数据由内核写入,存储内核检测之后的结果;
nfds :这是第一个参数数组中最后一个元素的下标 + 1,用来表示循环结束的条件;
timeout :指定 poll() 函数的阻塞时长:
- − 1 -1 −1,一直阻塞,直到检测的集合中有就绪的文件描述符才会解除阻塞;
- 0 0 0,不阻塞,不管待检测的集合中有没有文件描述符,函数执行完之后就返回;
- > 0 > 0 >0,阻塞指定的毫秒数,就解除阻塞返回;
函数返回值:
- − 1 -1 −1,失败;
- > 0 >0 >0,成功。返回的数就是集合中已经就绪的文件描述符的总个数;
2.实现
客户端代码:
#include <stdio.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <string.h>
int main(){
    //1.创建用于通信的文件描述符 cfd
    int cfd = socket(AF_INET,SOCK_STREAM,0);
    if(cfd == -1){
        perror("socket");
        return -1;
    }
    printf("1.成功创建了用于通信的文件描述符 : %d\n",cfd);
    //2.连接服务器
    unsigned short port = 10000;
    const char* ip = "10.0.8.14";
    struct sockaddr_in addr;
    addr.sin_family = AF_INET;
    addr.sin_port = htons(port);
    inet_pton(AF_INET,ip,&addr.sin_addr.s_addr);
    int ret = connect(cfd,(struct sockaddr*)&addr,sizeof(addr));
    if(ret == -1){
        perror("connet");
        return -1;
    }
    printf("2.成功连接了服务器 , ip : %s , port : %d\n",ip,port);
    //3.开始通信
    char send_buf[1024];
    char recv_buf[1024];
    int cnt = 0;
    while(1){
        memset(send_buf,0,sizeof send_buf);
        memset(recv_buf,0,sizeof recv_buf);
        sprintf(send_buf,"hello i love you : %d",cnt++);
        //发送数据
        send(cfd,send_buf,strlen(send_buf) + 1,0);
        //接收数据
        int len = recv(cfd,recv_buf,sizeof(recv_buf),0);
        if(len > 0){
            printf("服务端 : %s\n",recv_buf);
        }
        else if(len == 0){
            printf("服务端已经断开了连接...\n");
            break;
        }
        else{
            perror("recv");
            break;
        }
        sleep(1);
    }
    close(cfd);
    return 0;
}
服务端代码:
#include <stdio.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <pthread.h>
#include <strings.h>
#include <string.h>
#include <stdlib.h>
#include <poll.h>
int main()
{
    // 1.创建用于监听的文件描述符
    int lfd = socket(AF_INET, SOCK_STREAM, 0);
    if (lfd == -1)
    {
        perror("socket");
        return -1;
    }
    // 2.绑定 ip 和 端口号
    unsigned short port = 10000;
    struct sockaddr_in saddr;
    saddr.sin_family = AF_INET;
    saddr.sin_port = htons(port);
    saddr.sin_addr.s_addr = INADDR_ANY;
    int ret = bind(lfd, (struct sockaddr *)&saddr, sizeof(saddr));
    if (ret == -1)
    {
        perror("bind");
        return -1;
    }
    // 3.设置监听
    ret = listen(lfd, 128);
    if (ret == -1)
    {
        perror("listen");
        return -1;
    }
    printf("设置监听成功...\n");
    // 4.获取连接
    struct pollfd fds[1024];
    for (int i = 0; i < 1024; i++)
    {
        fds[i].fd = -1;
        fds[i].events |= POLLIN;
    }
    // 把用于监听的文件描述符 lfd 加入到 fds 数组中
    fds[0].fd = lfd;
    int addr_len = sizeof(struct sockaddr_in);
    int maxfd = 0;
    char buf[1024];
    char *str = "ok";
    while (1)
    {
        int ret = poll(fds, maxfd + 1, -1);
        if(ret == -1){
            perror("poll");
            exit(0);
        }
        // 检测 用于监听的文件描述符 lfd 读缓冲区是否有数据 , 也就是是否有新的连接到来
        if (fds[0].revents & POLLIN)
        {
            struct sockaddr_in addr;
            // 获取连接 , 返回用于通信的文件描述符
            int cfd = accept(lfd, (struct sockaddr *)&addr, &addr_len);
            printf("获取连接成功 , 客户端 ip : %s  , port : %d\n", inet_ntoa(addr.sin_addr), ntohs(addr.sin_port));
            // 把用于通信的文件描述符 cfd 加入到 fds 中
            int i = 1;
            for (; i < 1024; i++)
            {
                if (fds[i].fd == -1)
                {
                    fds[i].fd = cfd;
                    break;
                }
            }
            maxfd = i > maxfd ? i : maxfd;
        }
        for (int i = 1; i <= maxfd; i++)
        {
            // 用于通信的文件描述符 cfd 的读缓冲区有数据了 , 也就是有客户端发消息过来了
            if (fds[i].revents & POLLIN)
            {
                int cfd = fds[i].fd;
                memset(buf, 0, sizeof buf);
                int len = read(cfd, buf, sizeof buf);
                printf("客户端 : %s\n", buf);
                if (len > 0)
                {
                    write(cfd, str, strlen(str) + 1);
                }
                else if (len == 0)
                {
                    // 客户端已经关闭了连接
                    printf("客户端已经关闭了连接...\n");
                    close(cfd);
                    fds[i].fd = -1;
                }
                else
                {
                    perror("read");
                    exit(0);
                }
            }
        }
    }
    close(lfd);
    return 0;
    }
3.epoll
1.概念
epoll 全称是 eventpoll,是 Linux Io多路复用技术的一个实现之一。epoll 可以说是 select 和 poll 的升级版,相比于这两个,epoll 的底层数据结构是红黑树,实现起来更为高效。
- 对于待检测集合select和poll是基于线性方式处理的,epoll是基于红黑树来管理待检测集合的。
- select和poll每次都会线性扫描整个待检测集合,集合越大速度越慢,epoll使用的是回调机制,效率高,处理效率也不会随着检测集合的变大而下降
- select和poll工作过程中存在内核/用户空间数据的频繁拷贝问题;
- 程序猿需要对select和poll返回的集合进行判断才能知道哪些文件描述符是就绪的,通过epoll可以直接得到已就绪的文件描述符集合,无需再次检测
- 使用epoll没有最大文件描述符的限制,仅受系统中进程能打开的最大文件数目限制;
当 IO多路复用 的文件描述符数量庞大、IO流量频繁的时候,一般不太适合使用select() 和 poll() ,这种情况下 select() 和 poll() 表现较差,推荐使用 epoll() 。
2.函数原型
epoll 的三个 API 函数:
#include <sys/epoll.h>
// 创建epoll实例,通过一棵红黑树管理待检测集合
int epoll_create(int size);
// 管理红黑树上的文件描述符(添加、修改、删除)
int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);
// 检测epoll树中是否有就绪的文件描述符
int epoll_wait(int epfd, struct epoll_event * events, int maxevents, int timeout);
epoll_create() 是创建一个红黑树模型的实例,用于管理待检测的文件描述符的集合:
int epoll_create(int size);
- size:在- Linux 2.6.8版本后这个参数是被忽略的,只需要指定一个大于 0 0 0 的数值即可;



















