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

标签 sockets loopback

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

#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()无法检测socket是否可读。它只是超时。我认为 sockfd 应该是可读的,因为我只是向环回地址发送了一条消息。代码有什么问题吗? 谢谢!

最佳答案

if i change the server port from 0 to 5000, then sendto() can succeed. What is the reason ?

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

对于 udp 数据包,您始终应该指定非零目标端口作为 sendto() 的参数。

I think sockfd should be readable since i just send a message to the loopback address.

我不确定,但看起来你没有听环回。绑定(bind)时,您仅在 192.168.2.12 网络接口(interface)上进行绑定(bind)。您应该使用 INADDR_ANY 绑定(bind)所有接口(interface),包括环回。

关于sockets - 如何将 UDP 消息发送到环回地址,然后从中读取?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18934627/

相关文章:

python - 在运行的进程中刷新paramiko中的stdout

Java - 通过名称 DNS (?) 获取 IP 地址

javascript - 如何在 Loopback 4 中查看文件更改?

node.js - 检查用户是否登录(Strongloop 和 NodeJS)

仅当 Fiddler 运行时,启用环回的 UWP 应用才能成功到达本地主机服务

python - 在 Python 3 中使用套接字传输文件

sockets - 为什么发送时没有像 MSG_WAITALL 这样的标志?

java - 与目标 VM 断开连接,地址 : '127.0.0.1:51928' , 传输: 'socket'

node.js - 如何在环回中包含与远程方法结果的关系?

javascript - 无法在环回中更新模型