linux select多线程使用问题。

发布于 2022-09-12 03:21:27 字数 287 浏览 20 评论 0

1.linux 两个线程 select同一个socket句柄,监控该句柄的可读状态,内核会怎么处理?可读时会内核会先唤醒队列里的其中一个线程处理,如果该线程读了,是否就不会唤醒余下的线程?这样看在进程调度的过程中很有可能会引起惊群效应。
2.linux 两个线程 分别select不同的socket句柄,监控句柄的可读状态,内核会怎么处理?如果线程1的max_fd=3,线程2的max_fd=4。当fd=4可读时,应该不会有问题。当fd=3可读时,是否就会导致内核可能先会唤醒线程2?
3.正确做法是否就是所有的select都应该在一个线程上执行?

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(2

岁月苍老的讽刺 2022-09-19 03:21:27
man 2 select
SYNOPSIS
       #include <sys/select.h>
       int select(int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, struct timeval *timeout);

DESCRIPTION
       select() allows a program to monitor multiple file descriptors, waiting until one or more of the file descriptors become "ready" for some class of I/O operation (e.g., input possible). A file
       descriptor is considered ready if it is possible to perform a corresponding I/O operation (e.g., read(2), or a sufficiently small write(2)) without blocking.
       select() can monitor only file descriptors numbers that are less than FD_SETSIZE; poll(2) and epoll(7) do not have this limitation. See BUGS.

可以观察到每次调用 select,都是传送要读写的 fd 集合给 内核,并阻塞等待内核响应。

当有读写事件发生的时候,select阻塞解除,同时 readfds,writefds,exceptfds 会清空并填入就绪的 fd 列表。

不存在你说的 1,2情况。

更仔细的内容请阅读 select的文档。

   Arguments
       The arguments of select() are as follows:

       readfds
              The  file descriptors in this set are watched to see if they are ready for reading.
              A file descriptor is ready for reading if a read operation will not block; in  par‐
              ticular, a file descriptor is also ready on end-of-file.

              After select() has returned, readfds will be cleared of all file descriptors except
              for those that are ready for reading.

       writefds
              The file descriptors in this set are watched to see if they are ready for  writing.
              A  file  descriptor is ready for writing if a write operation will not block.  How‐
              ever, even if a file descriptor indicates as writable,  a  large  write  may  still
              block.

              After  select()  has returned, writefds will be cleared of all file descriptors ex‐
              cept for those that are ready for writing.

       exceptfds
              The file descriptors in this set are watched for "exceptional conditions".  For ex‐
              amples of some exceptional conditions, see the discussion of POLLPRI in poll(2).

              After  select() has returned, exceptfds will be cleared of all file descriptors ex‐
              cept for those for which an exceptional condition has occurred.
梦归所梦 2022-09-19 03:21:27
  1. 会唤醒多个线程

    简单示例:

    #include <stdio.h>
    #include <stdlib.h>
    #include <sys/select.h>
    #include <pthread.h>
    
    void *thread_start(void *arg) {
        int retval;
    
        fd_set rfds;
        struct timeval tv;
    
        /* Watch stdin (fd 0) to see when it has input. */
        FD_ZERO(&rfds);
        FD_SET(0, &rfds);
    
        /* Wait up to 30 seconds. */
        tv.tv_sec = 30;
        tv.tv_usec = 0;
    
        retval = select(1, &rfds, NULL, NULL, &tv);
        /* Don't rely on the value of tv now! */
    
        if (retval == -1)
            perror("select()");
        else if (retval)
            printf("Data is available now.\n"); // 两次
        else
            printf("No data within 30 seconds.\n");
    }
    
    int main(void) {
        pthread_t t1, t2;
        pthread_create(&t1, NULL, thread_start, NULL);
        pthread_create(&t2, NULL, thread_start, NULL);
        pthread_join(t1, NULL);
        pthread_join(t2, NULL);
    }
    

    更多分析可以看 Select is fundamentally broken

  2. 看两个 fdset 里的文件描述符是否有交集(结论见第一个问题),和 max_fd 没关系
  3. 是,或者每个线程有独立的不重复的 fd_set
~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文