2016-02-11 40 views
0

我只是想等我连接我的套接字后,看看我能写入它。 因此,我创建&连接我的套接字,并使用“选择”来完成这项工作,除非......不正确地分类。奇怪的行为与选择等待写入套接字

这里是我的代码:

int SocketToBox(char boxIp[INET6_ADDRSTRLEN], int port) 
{ 
    struct sockaddr_in boxAddress; 
    int    sock  = -1; 

    if ((sock = socket(AF_INET, SOCK_STREAM | SOCK_NONBLOCK, 0)) == -1) { 
     printf("%s : socket(AF_INET, SOCK_STREAM | SOCK_NONBLOCK, 0) : errno = '%s'.\n", __func__, strerror(errno)); 
     return (-1); 
    } 

    memset(&boxAddress, 0, sizeof(boxAddress)); 
    boxAddress.sin_family = AF_INET; 
    boxAddress.sin_port = htons(port); 
    if (inet_pton(AF_INET, boxIp, &boxAddress.sin_addr) != 1) { 
     printf("%s : inet_pton(AF_INET, boxIp<'%s'>, &boxAddress.sin_addr) : errno = '%s'.\n", __func__, boxIp, strerror(errno)); 
     close(sock); 
     return (-1); 
    } 

    if (connect(sock, (struct sockaddr *)&boxAddress, sizeof(boxAddress)) == -1) { 
     if (errno != EINPROGRESS) { 
      printf("%s : connect(sock<%d>, (struct sockaddr *)&boxAddress, sizeof(boxAddress)) : errno = '%s'.\n", __func__, sock, strerror(errno)); 
      close(sock); 
      return (-1); 
     } 
    } 
    return (sock); 
} 



int main(void) 
{ 
    int sock; 
    fd_set writeFdSet; 
    char *message = "Hello World !"; 

    if ((sock = SocketToBox("192.192.192.192", 2000)) == -1) { 
     printf("%s : SocketToBox(\"192.192.192.192\", 2000) : callstack.\n", __func__); 
     return (1); 
    } 

    FD_ZERO(&writeFdSet); 
    FD_SET(sock, &writeFdSet); 

    if (select(sock + 1, NULL, &writeFdSet, NULL, NULL) == -1) { 
     printf("%s : select(sock<%d> + 1, NULL, &writeFdSet, NULL, NULL) : errno = '%s'.", __func__, sock, strerror(errno)); 
     return (1); 
    } 

    if (FD_ISSET(sock, &writeFdSet)) { 
     printf("socket UP.\n"); 

     if (send(sock, message, strlen(message), MSG_DONTWAIT) == -1) { 
      printf("%s : send(sock<%d>, message<'%s'>, strlen(message)<%zu>, MSG_DONTWAIT) : errno '%s'.\n", __func__, sock, message, strlen(message), strerror(errno)); 
      return (1); 
     } 
     printf("Message sended.\n"); 
    } else { 
     printf("socket DOWN.\n"); 
    } 

    FD_ZERO(&writeFdSet); 
    if (sock == -1) { 
     close(sock); 
    } 
    return (0); 
} 

而且我的输出:

socket UP. 
main : send(sock<3>, message<'Hello World !'>, strlen(message)<13>, MSG_DONTWAIT) : errno 'Connection timed out'. 

我错过了什么? 选择返回1,我也不明白!为什么是1而不是0? 为什么我的套接字被标记为“可写”通过选择发送失败? 我可以确保你不能ping 192.192.192.192。

+0

我建议你从阅读[选择'手册页](http://man7.org/linux/man-pages/man2/select.2.html)开始,它应该解释这两种行为。 –

+0

谢谢......真的。我已经阅读了这个选择手册页,我不明白这个行为。 我会尝试以这种方式来解决我的问题:为什么一个套接字,我故意连接到什么都没有,标记为选择可写? –

+0

'SOCK_NONBLOCK'和'MSG_DONTWAIT'是非标准的(即非POSIX)。这是什么操作系统? –

回答

0

the select manual page

成功时,select()pselect()返回包含在三个文件中描述符的数量返回描述符集

这就是为什么select功能为您返回1:它返回在集合中设置一个描述符。


对于第二个问题,也从手动页面:

文件描述符被认为是准备好,如果它能够进行对应的I/O操作(例如,read(2)而不阻塞,或足够小的write(2))。

那些writefds将被观看,看是否有可用空间写(虽然大量写仍可块)

因此,所有select会告诉你的是,描述符内部缓冲区中有可用的空间,但它没有说什么有关数据是否可以实际发送。

+0

好的,谢谢。 我完全不明白,在人的第一次阅读,但它是有道理的。 操作系统只是说,他有一个缓冲区准备写入,而不是套接字连接。 猜我会找到另一种方式来知道一个插座是否真的连接。谢谢。 –

0

因为写入未连接的套接字不会阻塞 - 而这是唯一保证您的选择。