【问题标题】:How to read() from clients and a FIFO using select() in C如何在 C 中使用 select() 从客户端和 FIFO 读取()
【发布时间】:2016-04-19 02:23:35
【问题描述】:

read() 来自连接到服务器的客户端,同时,我让 select() 知道来自 FIFO 的数据。现在,当数据写入 FIFO 时,select 会将所有数据写入客户端,但会一直返回,就好像它“准备好读取”一样。所以下一次读取设置为-1errno == EAGAIN。它这样做直到达到 fdmax。不过它工作正常。

但是,为什么我不断收到EAGAIN?有没有更好的方法来处理这个?还是这是正确的方法?

注意:我传递了O_RDWR|O_NONBLOCK,因此它还可以保留客户端发送的read()ing 数据,而不仅仅是FIFO。

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

#define PORT "9034"
int main(void) {
    fd_set master, read_fds;
    int fdmax, listener, newfd, sbytes, yes=1, i, j, rv; 
    struct sockaddr_storage remoteaddr; // client address
    socklen_t addrlen;
    char buf[256] = {0}, remoteIP[INET6_ADDRSTRLEN];
    struct addrinfo hints, *ai;

    FD_ZERO(&master);
    FD_ZERO(&read_fds);
    int fifo;
    if ((mkfifo("/tmp/fifo", 0666)) < 0)
        perror(strerror(errno));
    if ((fifo = open("/tmp/fifo", O_RDWR|O_NONBLOCK)) < 0)  
        perror(strerror(errno));

    // get us a socket and bind it
    memset(&hints, 0, sizeof hints);
    hints.ai_family = AF_UNSPEC;
    hints.ai_socktype = SOCK_STREAM;
    hints.ai_flags = AI_PASSIVE;
    rv = getaddrinfo(NULL, PORT, &hints, &ai);
    listener = socket(ai->ai_family, ai->ai_socktype, ai->ai_protocol);
    setsockopt(listener, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(int));
    bind(listener, ai->ai_addr, ai->ai_addrlen);
    freeaddrinfo(ai);
    listen(listener, 10);

    FD_SET (fifo, &master);
    FD_SET(listener, &master);
    fdmax = listener;
    for (;;) {
        read_fds = master;
        if (select(fdmax + 1, &read_fds, NULL, NULL, NULL) == -1) exit(4);

        for (i = 0; i <= fdmax; i++) {
            if (FD_ISSET(i, &read_fds)) {
                if (i == listener) {
                    addrlen = sizeof remoteaddr;
                    newfd = accept(listener, (struct sockaddr * ) & remoteaddr, & addrlen);
                    FD_SET(newfd, & master);
                    if (newfd > fdmax)
                        fdmax = newfd;
                } else if (i != fifo) {
                    recv(i, buf, sizeof buf, 0); 
                }   
            }
            if (FD_ISSET(fifo, &read_fds)) {
                sbytes = read (fifo, buf, sizeof (buf));
                if (sbytes == -1 && errno == EAGAIN)
                    continue;
                for(j = 0; j <= fdmax; j++) {
                    // send to everyone!
                    if (FD_ISSET(j, &master)) {
                        if (j != listener && j != i && j != fifo) {
                            if (send(j, buf, sbytes, 0) == -1) {
                                perror("send");
                            }   
                        }   
                    }   
                }   
            }   
        }   
    }   
    return 0;
}

【问题讨论】:

  • fdmax = listener; 是可疑的。您实际上应该评估哪个更大。您需要检查每个recv() 的结果。你没有在插座盒中这样做。具体来说,它可能为零,这意味着对等方已断开连接,您应该关闭套接字并将其从read_fdsmaster 中删除。
  • 谢谢。为简单起见,我删除了很多检查。不过我会确保添加它。
  • 对于fdmax,如果有帮助,您可以查看this piece of code 检查进程的最大文件限制(以及尝试将其最大化到操作系统的硬限制) )。
  • @Myst 感谢您发布此内容。

标签: c sockets select mkfifo


【解决方案1】:
for (i = 0; i <= fdmax; i++) {

你在这里迭代read_fds

if (FD_ISSET(fifo, &read_fds)) {

在这里您正在测试fifo 是否可读。每次绕圈。第一次,你读了一些东西,然后发送;下一次循环,FIFO没有发生任何事情,但上面的条件仍然成立,所以你再读一遍,得到EAGAIN

这个if 块应该在循环的外面和下面。

【讨论】:

  • 我现在明白了,它完全解决了这个问题。谢谢!
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2013-10-04
  • 1970-01-01
  • 2010-10-26
  • 1970-01-01
相关资源
最近更新 更多