套接字:连接被拒绝

时间:2019-02-06 00:10:47

标签: c sockets

我制作了一个TCP客户端/服务器,可以运行一次。但是,当我尝试多次运行它时,出现以下错误:“连接被拒绝”。这是我的代码。

客户

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>

#include <sys/socket.h>
#include <sys/types.h>

#include <netinet/in.h>

#include <arpa/inet.h>

int main() {
    int net_socket = socket(AF_INET, SOCK_STREAM, 0);

    struct sockaddr_in net_addr;

    net_addr.sin_addr.s_addr = inet_addr("127.0.0.1");
    net_addr.sin_family = AF_INET;
    net_addr.sin_port = htons(3250);

    int connection_state = connect(net_socket, (struct sockaddr*)&net_addr, sizeof(net_addr));

    if (connection_state == 0) {
        printf("Connected with the server.\n");
    }
    else {
        printf("Connection with the server failed. [%s]\n", strerror(errno));
    }

    char net_message[256];

    recv(net_socket, net_message, sizeof(net_message), 0);

    close(net_socket);
    shutdown(net_socket, SHUT_RDWR);

    return 0;
}

服务器

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>

#include <sys/socket.h>
#include <sys/types.h>

#include <netinet/in.h>

#include <arpa/inet.h>

int main() {
    int net_socket = socket(AF_INET, SOCK_STREAM, 0);

    struct sockaddr_in net_addr;

    net_addr.sin_addr.s_addr = inet_addr("127.0.0.1");
    net_addr.sin_family = AF_INET;
    net_addr.sin_port = htons(3250);

    bind(net_socket, (struct sockaddr*)&net_addr, sizeof(net_addr));

    listen(net_socket, 1);

    struct sockaddr_in cl_addr;

    int cl_size = sizeof(cl_addr);

    int cl_socket = accept(net_socket, (struct sockaddr*)&cl_addr, (socklen_t*)&cl_size);

    int loop = 1;
    char net_message[256];

    send(cl_socket, net_message, sizeof(net_message), 0);

    close(net_socket);
    close(cl_socket);
    shutdown(net_socket, SHUT_RDWR);
    shutdown(cl_socket, SHUT_RDWR);

    return 0;
}

我一直试图通过多种方式解决问题,但我是新手。感谢您的提前帮助!

1 个答案:

答案 0 :(得分:3)

问题在于您的服务器缺少SO_REUSEADDR套接字选项,这导致bind()系统调用失败。原因是,在客户端从服务器断开连接之后,系统仍以TIME_WAIT状态知道该连接,以等待较晚的数据包。如果未设置上述套接字选项,则会导致bind()失败,并EADDRINUSE出现。

使用以下内容:

...
int one = 1;
setsockopt(net_socket, SOL_SOCKET, SO_REUSEADDR, &one, sizeof(one));

bind(net_socket, (struct sockaddr*)&net_addr, sizeof(net_addr));
...

执行错误检查所有系统调用!那会更早地向您显示问题,并为您节省很多时间。

更多问题:

  • 您要发送未初始化的缓冲区net_message
  • 在监听套接字上关闭完全是不明智的,因为未在其上建立任何连接(此处为shutdown(net_socket, SHUT_RDWR);
  • 关闭客户端套接字(如果有的话)必须在close(cl_socket);之前 执行。但是在那种情况下根本没有必要。