我有一个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应该是可读的,因为我只是向环回地址发送消息。代码有什么问题吗? 谢谢!
答案 0 :(得分:1)
如果我将服务器端口从0更改为5000,则sendto()可以成功。是什么原因?
UDP要求数据包具有大于零的特定源和目标端口。可以使用零端口的唯一情况是绑定调用;在这种情况下,套接字将绑定在一些空闲的非零端口上,来自该套接字的未来数据包将使用该数字作为src端口。
您始终应将非零目标端口指定为udp数据包的sendto()参数。
我认为sockfd应该是可读的,因为我只是向环回地址发送消息。
我不确定,但看起来你不听环回。绑定时,仅绑定192.168.2.12网络接口。您应该使用INADDR_ANY绑定所有接口,包括环回。