0

在 C 语言中,假设我有一个服务器应用程序。套接字已正确设置并已绑定到本地端口,并且已调用 listen() 函数而没有错误。

像这样的代码:

    //obtain a connection.
    int connection;
    struct sockaddr_storage sender;
    socklen_t addr_size = sizeof(sender);
    if((connection=accept(sockfd, (struct sockaddr*)&sender, &addr_size))==-1)
            {close(sockfd); return -1; }
    if(close(sockfd)==-1){
            close(connection);
            return -1;
    }
    else
            return connection;

成功获取连接文件描述符,通过该描述符可以从客户端读取数据。

我的应用程序要求服务器等待一段有限的时间,然后在没有建立连接的情况下关闭并执行其他操作。此外,最多接受一个连接。Accept() 是一个阻塞系统调用,但我想避免阻塞。

似乎有很多不同的方法可以解决这个问题:

  • 一个SE question建议使用 select 来解决这个问题,但是 select 是为一次监视许多文件句柄而设计的,在这里似乎有点过分了。

  • 另一个问题建议将套接字修改为非阻塞,然后在循环中检查是否建立了连接。这也可以,但看起来不太好,因为我确实想阻止 - 只是不是永远。

  • 在其他语言中,我似乎有人使用 SIGALARM 和 alarm() 命令来防止阻塞,但我的应用程序有多个线程,这似乎是一个容易出错的解决方案。

  • 在其他一些语言(Java?)中,我认为有一个超时参数可以传递给接受它来完成这个。这在这里会很理想,但我似乎无法在 C 中找到类似的套接字功能。

有没有更好的方法来解决这个问题?这似乎是其他人已经想出来的事情。如果没有更好的方法,上面提到的哪种解决方案最合适?我倾向于使套接字不阻塞,但这似乎会引入大量额外的麻烦。

4

1 回答 1

1

使用 select 时,请记住始终使用 fcntl() 使您的文件描述符非阻塞,否则客户端可能会启动连接,select 将返回,客户端可能会终止,并且您的服务器将在 accept() 中被永久阻塞。

成功接受后,如果您的程序以这种方式工作,您可能会再次将文件描述符转换为阻塞模式。

于 2012-11-15T19:50:00.170 回答