如何将UDP消息发送到环回地址然后从中读取?

时间:2013-09-21 15:55:52

标签: sockets loopback

我有一个c程序。它首先尝试将UDP消息发送到环回地址,然后从环回读取。 但首先sendto()函数失败,消息“ sendto失败:无效参数”。 代码如下:

#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <stdio.h>

int main(void)
{
    struct sockaddr_in servaddr;
    int sockfd = socket(AF_INET, SOCK_DGRAM, 0);

    bzero(&servaddr, sizeof(servaddr));

    struct in_addr addr;
    char dottedaddr[20];
    inet_aton("127.0.0.1", &addr);

        servaddr.sin_family      = AF_INET;
        servaddr.sin_addr.s_addr = addr.s_addr;
        servaddr.sin_port        = htons(0);

    struct sockaddr_in cliaddr;

        inet_aton("192.168.2.12", &addr);

        cliaddr.sin_family      = AF_INET;
        cliaddr.sin_addr.s_addr = addr.s_addr;
        cliaddr.sin_port        = htons(5000);


    if(bind(sockfd, (const struct sockaddr *)&cliaddr, sizeof(cliaddr)) == -1)
    {
        perror("bind failed");
        exit(1);
    }

    char buf[] = {'h', 'i', ' ', 'i', ' ', 'l', 'o', 'v', 'e', ' ', 'y', 'o', 'u', 0};


    if( sendto(sockfd, buf, sizeof(buf), 0, (const struct sockaddr *)&servaddr, sizeof(servaddr)) == -1)
    {
        perror("sendto fails");
        exit(2);
    }

    fd_set readFd;
    FD_ZERO(&readFd);
    FD_SET(sockfd, &readFd);
    struct timeval timeout;
    timeout.tv_sec = 5;
    timeout.tv_usec = 0;

    int ret = select(sockfd + 1, &readFd, NULL, NULL, &timeout);
    if(ret > 0)
    {
        if(FD_ISSET(sockfd, &readFd))
        {
            char buf2[21];
            struct sockaddr_in from;
            int len = sizeof(from);
            if(recvfrom(sockfd, buf2, sizeof(buf2), 0, (struct sockaddr *)&from, &len) == -1)
            {
                perror("recvfrom fails");
            }
        }
    }
    else if (ret == 0)
    {
        printf("select time out \n");
    }
    else
    {
        printf("select fails");
    }

}

如果我将服务器端口从0更改为5000,则sendto()可以成功。是什么原因?

第二个问题是,在服务器端口更改为5000之后,select()无法检测到套接字是否可读。它只是超时。我认为sockfd应该是可读的,因为我只是向环回地址发送消息。代码有什么问题吗? 谢谢!

1 个答案:

答案 0 :(得分:1)

  

如果我将服务器端口从0更改为5000,则sendto()可以成功。是什么原因?

UDP要求数据包具有大于零的特定源和目标端口。可以使用零端口的唯一情况是绑定调用;在这种情况下,套接字将绑定在一些空闲的非零端口上,来自该套接字的未来数据包将使用该数字作为src端口。

您始终应将非零目标端口指定为udp数据包的sendto()参数。

  

我认为sockfd应该是可读的,因为我只是向环回地址发送消息。

我不确定,但看起来你不听环回。绑定时,仅绑定192.168.2.12网络接口。您应该使用INADDR_ANY绑定所有接口,包括环回。