- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我正在尝试创建一个 void mksockaddr(int af, int proto, char addr[], struct sockaddr* dst)
来创建一个 sockaddr 结构,这是我所做的:
void sockaddr(int af, int port, char addr[], struct sockaddr* dst) {
if (af == AF_INET) {
struct sockaddr_in s;
s.sin_family = af;
s.sin_port = htons(port);
inet_pton(af, addr, &s.sin_addr);
memcpy(dst, &s, sizeof(s));
} else {
struct sockaddr_in6 s;
s.sin6_family = af;
s.sin6_port = htons(port);
s.sin6_flowinfo = 0;
inet_pton(af, addr, &s.sin6_addr);
memcpy(dst, &s, sizeof(s));
}
}
这似乎对 AF_INET (IPv4) 没有问题,我可以 bind()
没有任何问题,但是当我尝试使用 AF_INET6 时,bind()
给出me 无效参数。
这是我用来bind()
的代码:
int sock_fd = socket(AF_INET6, SOCK_RAW, proto);
struct sockaddr sin;
sockaddr(AF_INET6, proto, src, &sin);
if(bind(sock_fd, &sin, sizeof(sin)) < 0) {
fprintf(stderr, "[ERR] can't bind socket: %s\n", strerror(errno));
exit(1);
} // got Invalid argument
但是,如果我自己构造一个 sockaddr_in6
,我可以 bind()
就好了:
struct sockaddr_in6 sin;
sin.sin6_port = htons(proto);
sin.sin6_family = AF_INET6;
inet_pton(AF_INET6, src, &sin.sin6_addr);
if(bind(sock_fd, (struct sockaddr*) &sin, sizeof(sin)) < 0) {
fprintf(stderr, "[ERR] can't bind socket.\n");
exit(1);
} // work just fine
所以我将函数创建的 sockaddr
转换回 sockaddr_in6
,我可以看到除了 sin6_scope_id
之外所有字段都是相同的。据我了解,sin6_scope_id
并不重要,除非我正在处理链接本地 IPv6 地址。
我在这里遗漏了什么吗?
最佳答案
从 C 的角度来看,为了确保您的代码按预期工作,调用者必须在 dst
参数中传递指向正确结构类型的有效指针。您的示例不会这样做。相反,它声明了一个 struct sockaddr
,并传递一个指向它的指针。类型 struct sockaddr
本身永远不会用作实际对象的类型,并且它对于所有可能的地址类型来说都不够大。特别是,它对于 IPv6 地址来说不够大。
另一方面,POSIX 比标准 C 对一致性程序的要求更快更宽松。这对于套接字地址尤其明显。它定义了一个类型 struct sockaddr_storage
来准确地满足您的目的:它足够大并且具有适当的对齐方式来保存任何支持的套接字地址类型的数据。文档特别提到了它在一般支持 IPv4 和 IPv6 方面的用途。 POSIX 还支持在不同的套接字地址指针类型之间进行强制转换,尽管这会导致违反 C 的结构别名规则。
因此,我将重写您的函数以明确使用 struct sockaddr_storage
,并且我将通过适当的转换进一步简化我的代码。此外,我会让我的函数告诉我地址结构的可用大小,它只包含初始化的那部分:
void populate_sockaddr(int af, int port, char addr[],
struct sockaddr_storage *dst, socklent_t *addrlen) {
if (af == AF_INET) {
struct sockaddr_in *dst_in4 = (struct sockaddr_in *) dst;
*addrlen = sizeof(*dst_in4);
memset(dst_in4, 0, *addrlen);
dst_in4->sin_family = af;
dst_in4->sin_port = htons(port);
inet_pton(af, addr, &dst_in4->sin_addr);
} else if (af == AF_INET6) {
struct sockaddr_in6 *dst_in6 = (struct sockaddr_in6 *) dst;
*addrlen = sizeof(*dst_in6);
memset(dst_in6, 0, *addrlen);
dst_in6->sin6_family = af;
dst_in6->sin6_port = htons(port);
// unnecessary because of the memset(): dst_in6->sin6_flowinfo = 0;
inet_pton(af, addr, &dst_in6->sin6_addr);
} // else ...
}
然后你可以像这样使用它:
struct sockaddr_strorage addr;
socklen_t addrlen;
populate_sockaddr(af, port, src, &addr, &addrlen);
if (bind(sock_fd, (struct sockaddr *) &addr, addrlen) < 0) {
// ...
}
请注意,将 &addr
转换为类型 struct sockaddr *
完全是例行公事。
关于C - 创建一个 sockaddr 结构,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48328708/
这个问题在这里已经有了答案: C++ Redefinition Header Files (winsock2.h) (15 个答案) 关闭 6 年前。 我使用 Visual Studio 2012
struct sockaddr { unsigned short sa_family; // address family, AF_xxx char sa_da
当我运行时 cc -c -o rawudp.o rawudp.c 我明白了 在 rawudp.c:1:0 包含的文件中: rawudp.h:48:23: 错误:未知类型名称‘sockaddr’ mak
我想知道确定使用哪个 sockaddr 变体的正确方法。 来自Beej's Guide to Network Programming : if (p->ai_family == AF_INET) {
我目前正在编写一个使用网络结构(例如 sockaddr)的 C 库。在我的代码中,我使用这个三元条件来推断这种结构的大小。 addrlen = sourceaddr->sa_family ==
在尝试以下操作时,第二个 sockaddr 中的地址发生了变化: /*Stuff*/ sockaddr add1, add2; recvfrom(/*socket*/, /*buffer*/, /*c
我正在尝试创建一个 void mksockaddr(int af, int proto, char addr[], struct sockaddr* dst) 来创建一个 sockaddr 结构,这是
我正在尝试学习网络编程,并在此过程中学习 C。我对作为通用地址的结构 sockaddr 和 sockaddr_in 感到困惑。我的书是这样说的: Thus, we can fill in the fi
根据msdn的这篇文章( http://msdn.microsoft.com/en-us/library/windows/desktop/ms740496(v=vs.85).aspx ) 结构因选择的
我正在打开一个 Linux 数据包套接字并尝试将接收到的数据包读入一个结构中: type msg struct { n, oobn, flags int p, oob []byte
我一直在阅读 Beej 的网络编程指南,在他的一个示例中,他将指向 struct sockaddr 的指针转换为 struct sockaddr_in6 指针,如下所示。 void *addr
'operator sockaddr *()'在这里是什么意思?类 Raw 是一个内部类,sockaddr 是一个结构。 struct sockaddr { __SOCKADDR_COMM
var sa_tab:[sockaddr?] = [sockaddr](repeating: sockaddr(), count: Int(RTAX_MAX)) let addr:sockaddr =
我遇到过这段代码: (stuct sockadrr*)&servaddr 这是什么意思?您以某种方式将结构强制转换为 servaddr? 我只是想弄清楚这段代码的语法及其在 C 中的含义。 最佳答案
概览 我正在用 C++ 编写一个简单的测试游戏服务器。目前,我有一个监听任何人的绑定(bind)套接字,当某种数据包类型到达时,我想将 sockaddr 结构保存到一个对象中,该对象将传递给一个单独的
我正在编写一个从命令行提供地址的 UDP 套接字程序。 为了使发送和写入更容易,我使用 getaddrinfo 将地址转换为 sockaddr 结构:sockaddr_in 或 sockaddr_in
手册页未涵盖在发生错误时通过引用传递到 accept() 函数的 sockaddr 变量的状态。 是否可以安全地假设如果在客户端连接和您接受它之间出现问题,导致 accept() 的返回值小于 0,那
目前我正在尝试部署一个用C++编写的网络应用程序,由于网络问题,它看起来陷入了死循环: [isaev@feck-5 ~]$ sudo strace -p 26252 -f -e trace=netwo
我用 C 编写了一个简短的 TCP 服务器示例,它持续监听端口 12701 上的连接,并将对等方的 sockaddr.sa_family 的值输出到标准输出。程序在无限循环中调用 accept(),它
我在 Mac OS X 中做了一些干预(本质上是拦截 C 调用),我注意到 ping 应用程序试图调用 addrlen 值为 16 的 sendto 函数。在 sys/socket.h 中我可以清楚地
我是一名优秀的程序员,十分优秀!