我正在编写一个服务器客户端程序。 在服务器上,我通过以下数据结构的数组管理客户端:
struct Client
{
struct sockaddr_in addr;
/*...*/
};
struct Client CLIENTS[MAX_CLI];
当我通过 UDP
套接字从客户端收到第一个数据包时
struct sockaddr_in addr_cli;
memset(&addr_cli,0,sizeof(struct sockaddr_in));
b=recvfrom(SOCK_UDP_FATHER, &pdu, sizeof(pdu), MSG_DONTWAIT,
(struct sockaddr *)&addr_cli, (socklen_t *)&laddr_cli);
我想将他的地址复制到我的结构中。所以我这样做:
memcpy(&CLIENTS[client].addr,(struct sockaddr*)&addr_cli,
sizeof(struct sockaddr_in));
printf("IP client: %s",inet_ntoa(CLIENTS[client].addr.sin_addr);
奇怪的是,第一次尝试通信失败,打印0.0.0.0
。但客户的下一次尝试是成功的,一切都很顺利。为什么会发生这种情况?
最佳答案
调用 recvfrom
未填充 addr_cli
。 recvfrom
的最后两个参数有点棘手。他们是
struct sockaddr *src_addr,
socklen_t *addrlen
如果src_addr
不为NULL,则recvfrom
期望addrlen
指向src_addr
缓冲区的长度。 (通常是sockaddr
结构的大小)。如果addrlen
的值太小,返回的地址将被截断。调用 recvfrom
后,addrlen
将被设置为实际地址长度。看documentation了解详情。
所以你需要在调用recvfrom()
之前初始化laddr_cli
:
struct sockaddr_in addr_cli;
socklen_t laddr_cli = sizeof(addr_cli); // <--
memset(&addr_cli,0,laddr_cli);
b=recvfrom(SOCK_UDP_FATHER, &pdu, sizeof(pdu), MSG_DONTWAIT,
(struct sockaddr *)&addr_cli, &laddr_cli);
关于c - 如何 memcpy() struct sockaddr_in,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29398634/