1、概念
基于IPv4的socket网络编程,sockaddr_in中的成员struct in_addr sin_addr表示32位 的IP 地址。网络字节顺序的IP地址是二进制的数据,但是通常用点分⼗进制的字符串表⽰IP 地址,为了方便使用需要转换为点分十进制的字符串。例如:128.2.194.242就是地址0x8002c2f2的点分十进制表示。应用程序可以使用以下库函数实现IP地址与点分十进制串的转换,它们存放在<arpa/inet.h>头文件中。
2、IP地址转换函数
(1)int inet_pton(int doman, const char *src, void *dst);
释:该函数将一个点分十进制串转换为一个二进制的网络字节顺序的IP地址。如果src没有指向一个合法的点分十进制字符串,那么该函数返回0。成功返回1,失败返回-1。
(2)const char* inet_ntop(int doman, constnvoid* src, char* dst, socklen_t size);
释:该函数将一个二进制的网络字节顺序的IP地址转换为它对应的点分十进制的字符串,并把得到的以null结尾的字符串复制到dst。成功返回指向点分十进制的指针,失败返回NULL。
【例1】
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<arpa/inet.h>
int main()
{
char buf[1024] = {0};
uint32_t *ptr = NULL;
int ret = 0;
socklen_t len = 0;
struct sockaddr_in addr;
ret = inet_pton(AF_INET, "127.0.0.1", &addr.sin_addr);
if(ret == 0)
{
perror("enter IP error!\n");
exit(-1);
}
else if(ret == -1)
{
perror("inet_pton error!\n");
exit(-1);
}
ptr = (uint32_t *)(&addr.sin_addr);
len = sizeof(struct sockaddr_in);
printf("ptr:%x\n", ptr);
printf("addr:%s\n", inet_ntop(AF_INET, &addr.sin_addr, buf, len));
return 0;
}
运行结果:
(3)int inet_aton(const char *string, struct in_addr* addr);
释:输入参数string包含ASCII表示的IP地址, 输出参数addr是将要用新的IP地址更新的结构。如果输入地址不正确,则返回0;如果成功,返回非零;如果失败,返回-1。
(4)char *inet_ntoa(struct in_addr in);
释:该函数在内部申请了一块空间保存返回的点分十进制的IP地址。因为返回的结果放到静态存储区,所以不需要手动释放。
【例2】
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<arpa/inet.h>
int main()
{
struct sockaddr_in addr1;
struct sockaddr_in addr2;
addr1.sin_addr.s_addr = 0;
addr2.sin_addr.s_addr = 0xfffffff;
char* ptr1 = inet_ntoa(addr1.sin_addr);
char* ptr2 = inet_ntoa(addr2.sin_addr);
printf("ptr1:%s, ptr2:%s\n", ptr1, ptr2);
return 0;
}
运行结果:
释:因为inet_ntoa把结果放到内部的一个静态存储区,这样第二次调用时的结果会覆盖掉上一次的结果。
注意:
1)在APUE中,明确提出inet_ntoa不是线程安全的函数。
2)在多线程环境下,推荐使用inet_ntop,这个函数由调用者提供一个缓冲区保存结果,可以规避线程安全问题。
(5)in_addr_t inet_addr(const char *str);
释:该函数可将点分十进制的IP地址转换为无符号长整型。