Розетки: соединение отказано - PullRequest
0 голосов
/ 06 февраля 2019

Я сделал TCP Client / Server, и я могу запустить его один раз.Но когда я пытаюсь запустить его более одного раза, выдается следующая ошибка: «Отказано в соединении».Вот мой код.

Клиент

#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 голосов
/ 06 февраля 2019

Проблема в том, что на вашем сервере отсутствует опция сокета 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));
...

и do проверка ошибок для всех системных вызовов!Это показало бы вам проблему намного раньше и сэкономило бы вам много времени.

Еще несколько проблем:

  • Вы отправляете неинициализированный буфер net_message
  • Завершение работы на слушающем сокете вообще не имеет смысла, поскольку на нем не установлено никакого соединения (здесь: shutdown(net_socket, SHUT_RDWR);)
  • Завершение работы на клиентском сокете - если вообще - должно быть выполнено до close(cl_socket);.Но в этом случае это совсем не обязательно.
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...