socket select函数用法,图文并茂讲解,初学者必备

目录

1.函数介绍

1.1 select函数

1.2 位图操作函数

 2.select模型 

3.select示例

3.1 服务端程序

3.2 客户端程序

4.重点和难点分析

4.1 如何正确设置读,写,异常集合(位图)?

4.2 如何正确设置超时时间?


1.函数介绍

1.1 select函数

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

int select(int nfds, fd_set *readfds, fd_set *writefds,          
          fd_set *exceptfds, struct timeval *timeout);

参数:
nfds:最大文件描述符+1
readfds:读文件描述符集合,可设置为NULL
writefds:写文件描述符集合,可设置为NULL
exceptfds:异常文件描述符集合,可设置为NULL
timeout:超时时间,设置为NULL为阻塞模式

返回值:
成功:返回检测到的文件描述符数量
失败:返回-1,设置errno
超时:返回0


1.2 位图操作函数

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

//设置fd对应位图位置为0
void FD_CLR(int fd, fd_set *set); 
//判断fd对应位图位置是否为1
int FD_ISSET(int fd, fd_set *set); 
//设置fd对应位图位置为1
void FD_SET(int fd, fd_set *set);
//整个位图清零
void FD_ZERO(fd_set *set);


select位图工作原理

图 1

 
2.select模型 

图 2


3.select示例

3.1 服务端程序

#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>
#include <fcntl.h>
#include <sys/select.h>
#include <sys/time.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

#define LISTEN_BACKLOG (5)
#define BUF_SIZE (1500)

#define ACK_STR "ack ok"

void usage(void) {
    printf("*********************************\n");
    printf("./server 本端ip 本端端口\n");
    printf("*********************************\n");
}

int main(int argc, char *argv[])
{
    struct sockaddr_in local;
    struct sockaddr_in peer;
    int sock_fd = 0, new_fd = 0;
    int ret = 0;
    socklen_t addrlen = 0;
    char send_buf[BUF_SIZE] = {0};
    char recv_buf[BUF_SIZE] = {0};
    
    if (argc != 3) {
        usage();
        return -1;
    }
    
    char *ip = argv[1];
    unsigned short port = atoi(argv[2]);
    printf("ip:port->%s:%u\n", argv[1], port);
    
    sock_fd = socket(AF_INET, SOCK_STREAM, 0);
    if (sock_fd == -1) {
        perror("socket error");
        return -1;
    }
    
    memset(&local, 0, sizeof(struct sockaddr_in));
    local.sin_family = AF_INET;
    local.sin_addr.s_addr = inet_addr(ip);
    local.sin_port = htons(port);
    
    ret = bind(sock_fd, (struct sockaddr *)&local, sizeof(struct sockaddr));
    if (ret == -1) {
        close(sock_fd);
        perror("bind error");
        return -1;
    }
    
    ret = listen(sock_fd, LISTEN_BACKLOG);
    if (ret == -1) {
        close(sock_fd);
        perror("listen error");
        return -1;
    }
    
    fd_set rfds;
    fd_set rfds_storage;
    FD_ZERO(&rfds_storage);
    FD_SET(sock_fd, &rfds_storage);
    
    int max_fd = sock_fd;
    while (1) {
        rfds = rfds_storage;
        struct  timeval tv = {.tv_sec = 5, .tv_usec = 0};
        ret = select(max_fd + 1, &rfds, NULL, NULL, &tv);
        if (ret == -1) {
            perror("select error");
            break;
        } else if (ret == 0) {
            printf("select timeout\n");
            continue;
        } else {
            printf("select ok\n");
        }
        
        for (int fd = 0; fd < max_fd + 1; fd++) {
            if (FD_ISSET(fd, &rfds)) {
                if (fd == sock_fd) {
                    addrlen = sizeof(peer);
                    new_fd = accept(sock_fd, (struct sockaddr *)&peer, &addrlen);
                    if (new_fd == -1) {
                        perror("accept error");
                        continue;
                    }
                    FD_SET(new_fd, &rfds_storage);
                    max_fd = (new_fd <= max_fd) ? max_fd : new_fd;
                    printf("accept success new fd:%d\n", new_fd);
                } else {
                    memset(recv_buf, 0, BUF_SIZE);
                    ret = recv(fd, recv_buf, BUF_SIZE, 0);
                    if (ret <= 0) {
                        close(fd);
                        FD_CLR(fd, &rfds_storage);
                    } else {
                        printf("recv len:%d, %s\n", ret, recv_buf);
                    }
                }
            }
        }
    }
    
    for (int fd = 0; fd < max_fd + 1; fd++) {
        if (FD_ISSET(fd, &rfds)) {
            printf("clsoe fd:%d\n", fd);
            close(fd);
        }
    }
    
    FD_ZERO(&rfds);
    FD_ZERO(&rfds_storage);
    
    return 0;
}

3.2 客户端程序

#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

#define LISTEN_BACKLOG (5)
#define BUF_SIZE (1500)

#define REQUEST_STR "tcp pack"

void usage(void) {
    printf("*********************************\n");
    printf("./client 对端ip 对端端口\n");
    printf("*********************************\n");
}

int main(int argc, char *argv[])
{
    struct sockaddr_in client;
    struct sockaddr_in server;
    int sock_fd = 0;
    int ret = 0;
    socklen_t addrlen = 0;
    char send_buf[BUF_SIZE] = {0};
    char recv_buf[BUF_SIZE] = {0};
    
    if (argc != 3) {
         usage();
         return -1;
     }
        
    char *ip = argv[1];
    unsigned short port = atoi(argv[2]);
    printf("ip:port->%s:%u\n", argv[1], port);
    
    sock_fd = socket(AF_INET, SOCK_STREAM, 0);
    if (sock_fd == -1) {
        perror("socket error");
        return -1;
     }
    memset(&server, 0, sizeof(struct sockaddr_in));
    server.sin_family = AF_INET;
    server.sin_addr.s_addr = inet_addr(ip);
    server.sin_port = htons(port);
    
    ret = connect(sock_fd, (struct sockaddr *)&server, 
                  sizeof(struct sockaddr));
    if (ret == -1) {
        close(sock_fd);
        perror("connect error");
        return -1;
    }
            
    int seq = 0;
    while(1) {
        memset(send_buf, 0, BUF_SIZE);
        sprintf(send_buf, "%s:%d", REQUEST_STR, seq++);
        send(sock_fd, send_buf, strlen(send_buf), 0);
        printf("send %s\n", send_buf);
        sleep(1);
    }
    
    close(sock_fd);
    return 0;
}


4.重点和难点分析

4.1 如何正确设置读,写,异常集合(位图)?

        每次select调用之前,必须重新设置读,写,异常文件描述符集合。select调用后,读,写,异常文件描述符位图会被内核修改,可分以下两种情况讨论:

        情况1:select超时后,会清空读,写,异常文件描述符集合,如果未重新设置读,写,异常文件描述符集合,将无法再次获取文件描述符读,写,异状态,导致无法读写数据。

        情况2:select某个文件描述符,因为读,写,异常没有变化,select会清空对应的位图,如果该文件描述符没有在select之前重新设置读,写,异常文件描述符集合,将无法再次获取该文件描述符读,写,异常状态,导致无法读写数据。


4.2 如何正确设置超时时间?

        每次select调用之前,需要重新设置超时时间。超时时间随着每次select之后,会相应的减少,直至减少至零,此时select超时失效,所以需每次select之前重新设置超时时间,才能确保能select正常超时。

  • 4
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
在 Python 中,可以使用 `select` 模块来实现基于事件驱动的 socket 编程。`select` 允许你监视多个 socket 对象,并在有数据可读、可写或发生错误时进行相应的处理。 下面是使用 `select` 的基本步骤: 1. 导入 `select` 模块:`import select` 2. 创建一个空的 `inputs` 列表,用于存储需要监视的 socket 对象:`inputs = []` 3. 将需要监视的 socket 对象添加到 `inputs` 列表中:`inputs.append(socket_obj)` 4. 使用 `select.select()` 函数来进行监视和等待事件的发生。该函数接受三个参数:`rlist` (可读对象列表)、`wlist` (可写对象列表) 和 `xlist` (错误对象列表)。在这里,我们通常只关心可读对象,所以将 `rlist` 设置为 `inputs`:`readable, writable, exceptional = select.select(inputs, [], [])` 5. 检查返回的 `readable` 列表,它包含了有数据可读的 socket 对象。可以使用循环遍历列表并处理每个可读对象: ``` for sock in readable: if sock == socket_obj: # 从 socket 接收数据 data = sock.recv(1024) # 处理接收到的数据 ``` 6. 重复步骤 4 和步骤 5,以便持续监视和处理事件。 需要注意的是,在使用 `select` 时,需要确保 socket 对象是非阻塞模式的,以避免阻塞整个程序。可以通过设置 `socket_obj.setblocking(False)` 来将 socket 对象设置为非阻塞模式。 这只是一个简单的示例,实际使用中可能需要更复杂的逻辑来处理不同的事件和错误。你可以根据具体需求进行扩展和定制。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

物联网心球

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值