我正在将IPv4应用程序移植到独立于 AF 的代码库(它应该与 IPv4 和 IPv6 一起使用)。现在我尽可能使用sockaddr_storage ,但是现在我必须设置(填充)一个sockaddr_storage。但我不知道正确的方法是什么。之前的代码是:
// defined in data_socket.h
struct sockaddr_in laddr;
现在有一个设置sin_addr和sin_port的函数:
void DataSocket::SetLocalAddr(const char *addr, const int port)
{
this->laddr.sin_port = htons(port);
if(addr != NULL)
this->laddr.sin_addr.s_addr = inet_addr(addr);
else
this->laddr.sin_addr.s_addr = inet_addr("0.0.0.0");
}
如您所见,这是旧样式(使用 IPv4)。
现在我的更改如下。首先,我已更改sockaddr_in
为sockaddr_storage
// defined in data_socket.h
struct sockaddr_storage laddr;
然后我更改了上面的代码以支持IPv4和IPv6:
void DataSocket::SetLocalAddr(const char *addr, const int port)
{
switch (this->GetAddrFamily(addr)) {
case AF_INET:
(struct sockaddr_in *) this->laddr.sin_port = htons(port);
if(addr != NULL)
inet_pton(AF_INET, addr, (struct sockaddr_in *) this->laddr.sin_addr);
else
inet_pton(AF_INET, "0.0.0.0", (struct sockaddr_in *) this->laddr.sin_addr);
break;
case AF_INET6:
(struct sockaddr_in6 *) this->laddr.sin6_port = htons(port);
if(addr != NULL)
inet_pton(AF_INET6, addr, (struct sockaddr_in6 *) this->laddr.sin6_addr);
else
inet_pton(AF_INET6, "0:0:0:0:0:0:0:0", (struct sockaddr_in6 *) this->laddr.sin6_addr);
break;
default:
return NULL;
}
}
在哪里GetAddrFamily()
:
int DataSocket::GetAddrFamily(const char *addr)
{
struct addrinfo hints, *res;
int status, result;
memset(&hints, 0, sizeof(hints));
hints.ai_family = AF_UNSPEC;
hints.ai_socktype = SOCK_STREAM;
if ((status = getaddrinfo(addr, 0, &hints, &res)) != 0)
{
fprintf(stderr, "getaddrinfo: %s\n", gai_strerror(status));
return false;
}
result = res->ai_family; // This might be AF_INET, AF_INET6,etc..
freeaddrinfo(res); // We're done with res, free it up
return result;
}
看来我的方法是复杂的。这是正确的方法吗?因为我已将 sockaddr_in 更改为 sockaddr_storage,实际上我只是想要这个问题的反面:Getting IPV4 address from a sockaddr structure
我试图找到最好的解决方案,例如这里:http ://www.kame.net/newsletter/19980604/它说永远不要使用inet_ntop()
and ,但是其他一些(比如 Beej的inet_pton()
网络教程)说应该使用用于基于IPv6的应用程序。inet_ntop()
inet_pton()
我的实施方式是正确的还是应该改变它?