我正在使用以下程序来调试更大的问题。
现在如果我在OSX上运行相同的程序,getopt永远不会返回0或1,它总是返回我设置的optval的任何值!!
我的程序中肯定有一些明显错误的东西正在逃避我。
示例我已将此程序运行为./a.out 64.233.160.105
./a.out 64.233.160.105
getsockopt: Undefined error: 0
getsockopt returned 0 SO_REUSEPORT -> 199
setsockopt reuseport: Undefined error: 0
set ret = 0 SO_REUSEPORT -> 1
setsockopt linger: Undefined error: 0
set ret = 0 SO_LINGER -> 0
getsockopt: Undefined error: 0
getsockopt returned 0 SO_REUSEPORT -> 99
sendto: Undefined error: 0
send to sent 11 bytes
以下是我正在使用的示例程序。
/* Sample UDP client */
#include <sys/socket.h>
#include <netinet/in.h>
#include <stdio.h>
#include <unistd.h>
#include <errno.h>
#include <stdlib.h>
#include <memory.h>
#include <arpa/inet.h>
int main(int argc, char **argv)
{
int sockfd, n;
struct sockaddr_in servaddr, cliaddr;
char *sendline = "Hello World";
char recvline[1000];
int ret;
extern int errno;
uint32_t optval;
socklen_t optlen;
struct linger linger_opt;
if (argc != 2) {
printf("usage: udpcli <IP address>\n");
exit(1);
}
sockfd = socket(AF_INET, SOCK_DGRAM, 0);
bzero(&servaddr, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr = inet_addr(argv[1]);
servaddr.sin_port = htons(4500);
errno = 0;
optval = 199;
ret = getsockopt(sockfd, SOL_SOCKET, SO_REUSEPORT, &optval,
(socklen_t *) & optlen);
perror("getsockopt");
printf("getsockopt returned %d SO_REUSEPORT -> %d\n", ret, optval);
errno = 0;
optval = 1;
ret = setsockopt(sockfd, SOL_SOCKET, SO_REUSEPORT, &optval, sizeof(optval));
perror("setsockopt reuseport");
printf("set ret = %d SO_REUSEPORT -> %d\n", ret, optval);
errno = 0;
optval = 0;
linger_opt.l_onoff = 0;
linger_opt.l_linger = 0;
ret = setsockopt(sockfd, SOL_SOCKET, SO_LINGER, &linger_opt, sizeof(linger_opt));
perror("setsockopt linger");
printf("set ret = %d SO_LINGER -> %d\n", ret, optval);
optval = 99;
errno = 0;
ret = getsockopt(sockfd, SOL_SOCKET, SO_REUSEPORT, &optval, &optlen);
perror("getsockopt");
printf("getsockopt returned %d SO_REUSEPORT -> %d\n", ret, optval);
errno = 0;
ret = sendto(sockfd, sendline, strlen(sendline), 0,
(struct sockaddr *) &servaddr, sizeof(servaddr));
perror("sendto");
printf("send to sent %d bytes \n",ret);
close(sockfd);
}
答案 0 :(得分:2)
最近我遇到了同样的问题。
错误: 罪魁祸首是&#34; sizeof&#34;。 Sizeof是一个运算符,它返回一个常量int,给出任何给定数据元素的大小.setsockopt&#39; socklen_t * optlen&#39;需要socklen_t类型的值。尽管socklen_t和int大多数时候都可以互换使用。这种使用可能导致难以追踪的错误,就像在这里一样。
ret = setsockopt(sockfd,SOL_SOCKET,SO_REUSEPORT,&amp; optval,sizeof(optval));
即使setsockopt没有返回错误或失败,由于使用&#39; sizeof(optval)&#39;,套接字选项实际上从未设置过。这就是为什么getsockopt无法返回更改后的值。
修复: 1.将optvalue的大小分配给optlen。 optlen = sizeof(optval); 2.使用optlen作为第5个参数设置套接字选项。 ret = setsockopt(sockfd,SOL_SOCKET,SO_REUSEPORT,&amp; optval,optlen);
这个修复在Linux上使用gcc编译器。希望它能帮助别人。