背景
Linux下这2个IP地址转换函数,可以在将IP地址在点分十进制字符串
和二进制结构体
之间转换。
函数名中的p和n非别代表表达(presentation)和数值(numeric)。地址的表达格式通常是ASCII字符串,数值格式则是存放到套接字地址结构中的二进制值。
inet_pton函数
#include<arpa/inet.h>
//字符串 -> 结构体
int inet_pton(int family, const char *strptr, void *addrptr);
返回:
1=成功
0=输入不是有效的表达式
-1=出错
inet_ntop函数
#include<arpa/inet.h>
//结构体 -> 字符串
const char *inet_ntop(int family, const void *addrptr, char *strptr, size_t len);
返回:若成功则为指向结果的指针(就是strptr值),若出错则为NULL。
入参:
family,AF_INET 或者 AF_INET6(如果以不被支持的地址族作为family的参数,这两个函数就都返回一个错误,并将errno置为EAFNOSUPPORT。)
strptr,字符串指针。
addrptr,指向二进制结构体。
len:存放字符串的大小,防止溢出。在<netinet/in.h>
中有定义:
#define INET_ADDRSTRLEN 16
#define INET6_ADDRSTRLEN 46
如果len太小,不足以容纳表达式结果(包括结尾的空字符),那么返回一个空指针,并置errno为ENOSPC。
v4例子
#include <stdio.h>
#include <netinet/in.h>
#include <arpa/inet.h>
int main() {
//ip字符串
char *ipStr = "1.2.3.4";
printf("ipStr=%s\n", ipStr);//1.2.3.4
// IPv4地址结构体(内存结构unsigned int)
struct in_addr s;
// 字符串 -> unsigned int
inet_pton(AF_INET, ipStr, &s.s_addr);
printf("s=0x%x\n", s.s_addr);//主机序,0x4030201
printf("htonl(s)=0x%x\n", htonl(s.s_addr));//网络序,0x1020304
//接收ip字符串的空间
char destIpStr[20] = {0};
// unsigned int -> 字符串
inet_ntop(AF_INET, &s.s_addr, destIpStr, 16);
printf("destIpStr=%s\n", destIpStr);//1.2.3.4
return 0;
}
v6例子
#include <stdio.h>
#include <netinet/in.h>
#include <arpa/inet.h>
int main() {
char *ipv6 = "2409:8a1e:6a62:e440:4f:bbe7:a27e:28e8";
struct in6_addr ip6;
char test[64] = {0};
inet_pton(AF_INET6, ipv6, &ip6);
printf("%x-%x\n", ip6.__in6_u.__u6_addr16[0], ip6.__in6_u.__u6_addr16[1]);//924-1e8a
inet_ntop(AF_INET6, &ip6, test, 64);
printf("test:%s\n", test);//2409:8a1e:6a62:e440:4f:bbe7:a27e:28e8
return 0;
}