Ошибка привязки при воссоздании сокета - PullRequest
11 голосов
/ 08 апреля 2011

A имеет следующий сокет слушателя:

int sd = socket(PF_INET, SOCK_STREAM, 0);

struct sockaddr_in addr;
bzero(&addr, sizeof(addr));
addr.sin_family = AF_INET;
addr.sin_port = htons(http_port);
addr.sin_addr.s_addr = INADDR_ANY;

if(bind(sd,(sockaddr*)&addr,sizeof(addr))!=0)
{
    ...
}

if (listen(sd, 16)!=0)
{
    ...
}

int sent = 0;
for(;;) {
    int client = accept(sd, (sockaddr*)&addr, (socklen_t*)&size);
    if (client > 0)
    {
        ...
        close(client);
    }
}

Если использовать

close(sd);

и затем пытаться воссоздать сокет с тем же кодом, возникает ошибка связывания, и только после30-60 секунд новый сокет успешно создан.

Есть ли способ создать или закрыть каким-нибудь классным способом, чтобы избежать ошибки связывания?

Ответы [ 4 ]

18 голосов
/ 08 апреля 2011

Где-то в ядре все еще есть информация о вашем предыдущем сокете. Скажите ядру, что вы все равно хотите повторно использовать порт:

int yes=1;
//char yes='1'; // use this under Solaris

if (setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(yes)) == -1) {
    perror("setsockopt");
    exit(1);
}

См. раздел bind () в Руководстве beej по сетевому программированию для более подробного объяснения.

2 голосов
/ 08 апреля 2011

Это ожидаемое поведение для сокетов TCP. Когда вы закрываете сокет, он переходит в состояние TIME_WAIT. Он будет принимать и отбрасывать пакеты для этого порта. Вам необходимо установить опцию SO_REUSEADDR для немедленной привязки.

1 голос
/ 08 апреля 2011

Вы не должны закрывать связанный сокет и затем пытаться воссоздать его.

accept возвращает только что созданный сокет только для этого соединения, это тот, который должен быть закрыт.то есть: вы должны делать -

close(client);
0 голосов
/ 08 апреля 2011

Попробуйте позвонить setsockopt с SO_REUSEADDR. См .: http://msdn.microsoft.com/en-us/library/ms740476(v=vs.85).aspx

...