IO多路复用技术(一)

IO多路复用

一、概念

IO多路复用技术 是一种 网络通信 的方式,通过这种方式可以同时检测多个 文件描述符这个过程是阻塞的),一旦检测到某一个文件描述符(状态是 可读 或者 可写 的)是就绪的,那么就会解除阻塞,然后我们就可以基于这些已经就绪的文件描述符进行网络通信了。

通过这种方式,服务端即使是在 单线程/进程 的情况下也能实现并发,支持多个连接。

常用的 IO多路复用 的方式有 : selectpollepoll

二、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 是跨平台的,同时支持 LinuxWindowsMacOS 。我们通过调用 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 0select() 函数就不会阻塞;

返回值:

  • 大于 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 234 个参数)需要频繁的在用户区和内核区之间进行数据的拷贝,效率低
  • 内核对于 select() 传递进来的待检测集合的检测方式是线性的
    • 如果集合内待检测的文件描述符很多,检测效率会比较低
    • 如果集合内待检测的文件描述符相对较少,检测效率会比较高
  • 使用 select 能够检测的最大文件描述符个数有上限,默认是 1024 1024 1024,这是在内核中被写死了的。

2.poll

poll 的机制跟 select 类似,使用方法也是类似的,以下是它们两个的对比:

  • 内核检测文件描述符的状态也是通过 线性遍历 的形式;
  • pollselect 检测的文件描述符的集合 会在被检测的过程频繁的进行 用户区 和 内核区的拷贝,它的开销随着文件描述符数量的增加而增大,所以效率也会变得越来越低;
  • select 可以跨平台使用,支持 LinuxWindowsMacOS;而 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;
    }
  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值