inet_aton、inet_addr、inet_ntoa、inet_pton、inet_ntop

inet_aton、inet_addr、inet_ntoa这3个函数是在点分十进制数串(例如“192.168.1.1”)与它长度为32位的网络字节序二进制值间转换IPV4地址。

两个较新的函数inet_pton、inet_ntop对于IPV4地址和IPV6地址都适用。

int inet_aton(const char *strptr, struct in_addr *addrptr);

/*
* 功能:strptr所指向的C字符串转换成一个32位的网络字节序二进制值,并通过指针addrptr来存储。
* 返回:若成功则返回1,否则返回0
*
* inet_aton函数有一个没写入正式文档中的特征:如果addrptr指针为空,那么该函数仍然对输入的字符串执行有效性检查,但是不存储任何结果。
*/
in_addr_t inet_addr(const char *strptr);

/*
* 功能:将strptr所指向的C字符串转换成一个32位的网络字节序二进制值,并返回该值。
* 返回:若字符串有效则返回一个32位二进制网络字节序的IPv4地址,否则返回INADDR_NONE
*
* inet_addr函数存在一个问题:所有2^32个可能的二进制值都是有效的IP地址(从0.0.0.0 到 255.255.255.255),但是当出错时该函数返回INADDR_NONE常值,该值通常是一个32位均为1的值。这就意味着点分十进制数串255.255.255.255(这是IPv4的有限广播地址)不能由该函数处理,因为它的二进制值被用来指示该函数失败。
* inet_addr函数还存在一个潜在的问题:一些手册声明该函数出错时返回-1而不是INADDR_NONE。这样在对该函数的返回值(一个无符号的值)与-1(一个负常值)进行比较时可能会发生问题,具体取决于C编译器。
*/
char* inet_ntoa(struct in_addr inaddr);

/*
* 功能:将一个32位的网络字节序二进制IPv4地址转换成相应的点分十进制数串。
* 返回:返回一个点分十进制数串的指针
*
* inet_aton函数的返回值所指向的字符串驻留在静态内存中,这就意味着该函数时不可重入的。
*/

/*
可重入函数和不可重入函数的概念:
  在函数中如果我们使用静态变量了,导致产生中断调用别的函数的 过程中可能还会调用这个函数,于是原来的 静态变量被在这里改变了,然后返回主体函数,用着的那个静态变量就被改变了,导致错误。这类函数我们称为不可重入函数。
  如果是在函数体内 动态申请内存的话,即便 新的线程调用这个函数也没事,因为新的线程使用的是新的函数的 新申请的动态内存(静态变量只有一份,所以 多线程对于函数体内的静态变量改变 会有无法修复的结果),所以这类函数就是可重入函数。
*/
int inet_pton(int family, const char *strptr, void *addrptr);

const char* inet_ntop(int family, const void *addrptr, char *strptr, size_t len);

/*
* 这两个函数的family参数既可以是AF_INET,也可以是AF_INET6。如果以不被支持的地址族作为family参数,这两个函数就都返回一个错误,并将errno置为EAFNOSUPPORT。
*
* inet_pton函数尝试转换由strptr指针所指的字符串,并通过addrptr指针存放二进制结果。若成功则返回1,否则如果对所指定的family而言输入的字符串不是有效的表达格式,那么返回0。
*
* inet_ntop函数将addrptr指针所指的二进制数值转换为字符串表达格式,并通过strptr存放字符串结果,len参数是目标存储单元的大小,以免该函数溢出其调用者的缓冲区。这个值通常是16(for IPv4点分十进制)或是46(for IPv6十六进制字符串)。如果len太小不足以容纳表达式结果(包括结尾的空字符),那么返回一个空指针,并置errno为ENOSPC。inet_ntop函数的strptr参数不可以是一个空字符串。调用者必须为目标存储单元分配内存并制定其大小。调用成功时,这个指针就是该函数的返回值。
*/

可以使用

inet_pton(AF_INET, cp, &foo.sin_addr);

替代

foo.sin_addr.s_addr = inet(cp);

可以使用

char str[INET_ADDRSTRLEN]; //ADDRSTRLEN = 16
ptr = inet_ntop(AF_INET, &foo.sin_addr, str, sizeof(str));

替代

ptr = inet_ntoa(foo.sin_addr);

 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
inet_ptoninet_ntop是用于IP地址转换的两个函数,分别用于将点分十进制的IPv4地址转换成网络字节序的二进制形式和将网络字节序的二进制形式的IPv4地址转换成点分十进制形式。 下面是它们的实现: ```c #include <arpa/inet.h> #include <string.h> int inet_pton(int af, const char *src, void *dst) { if (af == AF_INET) { struct in_addr addr; if (inet_aton(src, &addr) == 0) { return 0; } memcpy(dst, &addr.s_addr, sizeof(addr.s_addr)); return 1; } else if (af == AF_INET6) { struct in6_addr addr; if (inet_pton(AF_INET6, src, &addr) == 0) { return 0; } memcpy(dst, &addr, sizeof(addr)); return 1; } return -1; } const char *inet_ntop(int af, const void *src, char *dst, socklen_t size) { if (af == AF_INET) { struct in_addr addr; memcpy(&addr.s_addr, src, sizeof(addr.s_addr)); if (inet_ntop(AF_INET, &addr, dst, size) == NULL) { return NULL; } return dst; } else if (af == AF_INET6) { struct in6_addr addr; memcpy(&addr, src, sizeof(addr)); if (inet_ntop(AF_INET6, &addr, dst, size) == NULL) { return NULL; } return dst; } return NULL; } ``` 对于inet_pton函数,我们首先判断地址族af是否为IPv4或IPv6,然后根据不同的地址族调用inet_atoninet_pton进行转换。如果转换成功,则将转换后的二进制地址复制到dst指针所指向的内存中,并返回1;否则返回0表示转换失败。 对于inet_ntop函数,我们同样首先判断地址族af是否为IPv4或IPv6,然后根据不同的地址族调用inet_ntop进行转换。如果转换成功,则将转换后的点分十进制地址复制到dst指针所指向的内存中,并返回dst指针;否则返回NULL表示转换失败。需要注意的是,目标内存大小size应该足够存放转换后的地址。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值