Warning: file_get_contents(/data/phpspider/zhask/data//catemap/3/sockets/2.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Sockets 如何将UDP消息发送到环回地址,然后从中读取?_Sockets_Loopback - Fatal编程技术网

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

Sockets 如何将UDP消息发送到环回地址,然后从中读取?,sockets,loopback,Sockets,Loopback,我有一个c程序。它首先尝试将UDP消息发送到环回地址,然后从环回中读取。 但首先,sendto函数失败,消息为sendto fails:Invalid参数。 代码如下所示: #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <stdio.h> int main(void) { struct sockaddr_in servaddr

我有一个c程序。它首先尝试将UDP消息发送到环回地址,然后从环回中读取。 但首先,sendto函数失败,消息为sendto fails:Invalid参数。 代码如下所示:

#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应该是可读的,因为我只是向环回地址发送了一条消息。代码有什么问题吗? 谢谢大家!

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

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

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

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


我不确定,但看起来你好像不听回音。绑定时,仅在192.168.2.12网络接口上绑定。您应该使用INADDR\u ANY绑定所有接口,包括环回。

Hi,您找到解决方案了吗?我也不明白为什么服务器和客户端地址是这样的。。。