套接字编程 - setsockopt:协议不可用?

13

我在 C 语言中进行基本的 socket 编程,但在我尝试在每台电脑上运行代码时都遇到了这个错误。代码编译没有问题,但当我尝试运行时,会出现 setsockopt: Protocol not available 的错误。这似乎不是一个很常见的错误,但我在每台尝试运行的 MacOS 电脑上都遇到了它。

#include <unistd.h> 
#include <stdio.h> 
#include <sys/socket.h> 
#include <stdlib.h> 
#include <netinet/in.h> 
#include <string.h> 
#define PORT 8080 
int main(int argc, char const *argv[]) 
{ 
    int server_fd, new_socket, valread; 
    struct sockaddr_in address; 
    int opt = 1; 
    int addrlen = sizeof(address); 
    char buffer[1024] = {0}; 
    char *hello = "Hello from server"; 

    // Creating socket file descriptor 
    if ((server_fd = socket(AF_INET, SOCK_STREAM, 0)) == 0) 
    { 
        perror("socket failed"); 
        exit(EXIT_FAILURE); 
    } 

    // Forcefully attaching socket to the port 8080 
    if (setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR | SO_REUSEPORT, 
                                                  &opt, sizeof(opt))) 
    { 
        perror("setsockopt"); 
        exit(EXIT_FAILURE); 
    } 
    address.sin_family = AF_INET; 
    address.sin_addr.s_addr = INADDR_ANY; 
    address.sin_port = htons( PORT ); 

    // Forcefully attaching socket to the port 8080 
    if (bind(server_fd, (struct sockaddr *)&address,  
                                 sizeof(address))<0) 
    { 
        perror("bind failed"); 
        exit(EXIT_FAILURE); 
    } 
    if (listen(server_fd, 3) < 0) 
    { 
        perror("listen"); 
        exit(EXIT_FAILURE); 
    } 
    if ((new_socket = accept(server_fd, (struct sockaddr *)&address,  
                       (socklen_t*)&addrlen))<0) 
    { 
        perror("accept"); 
        exit(EXIT_FAILURE); 
    } 
    valread = read( new_socket , buffer, 1024); 
    printf("%s\n",buffer ); 
    send(new_socket , hello , strlen(hello) , 0 ); 
    printf("Hello message sent\n"); 
    return 0; 
} 

我被告知需要包含 setsockopt() 方法,以便在 TCP 连接关闭后等待时间之后重新启动的进程可以连接到地址而不被阻塞。有人知道如何解决这个错误吗?


7
你不能使用“|”符号组合套接字选项,因为它们不是位掩码。 - Barmar
7
您需要调用setsockopt()两次,一次用于SO_REUSEADDR,另一次用于SO_REUSEPORT - Barmar
实际上,您只需要设置 SO_REUSEADDR - user207421
请参考:https://dev59.com/rWYq5IYBdhLWcg3wjBQM - dash-o
非常感谢,我将SO_REUSEADDR和SO_REUSEPORT放到不同的setsockopt()方法中,现在它可以正常工作了。 - user8768055
1个回答

26

我曾遇到过类似的问题,只需要替换这段代码。

// Forcefully attaching socket to the port 8080 
if (setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR | SO_REUSEPORT, &opt, sizeof(opt)))

随着这个

// Forcefully attaching socket to the port 8080 
if (setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR , &opt, sizeof(opt)))

这对我来说很好用(在 macOS 上)。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接