注:下文的测试案例都是在Linux(Ubuntu 14.04)平台上测试通过
IP地址转换函数
在比较IP地址之前需要将其转换为网络字节序的二进制整数,常用的转换函数是 inet_pton()
int inet_pton(int af, const char *src, void *dst);
字符串IPv4地址转换实例
#include <arpa/inet.h>
int main()
{
char *str = "192.168.1.1";
struct in_addr ipv4; /* 保存IPv4地址的结构体 */
inet_pton(AF_INET, str, &ipv4); /* AF_INET表示IPv4地址协议族 */
return 0;
}
字符串IPv6地址转换实例
#include <arpa/inet.h>
int main()
{
char *str = "2000::1";
struct in6_addr ipv6; /* 保存IPv6地址的结构体 */
inet_pton(AF_INET6, str, &ipv6); /* AF_INET6表示IPv6地址协议族 */
return 0;
}
IP地址比较
原理
大端模式 - 数据的高字节保存在内存的低地址中
小端模式 - 数据的高字节保存在内存的高地址中
以IPv4地址"192.168.2.1"为例:
下面的例子,第一字节为最低地址,第四字节为最高地址
模式 | 第一字节 | 第二字节 | 第三字节 | 第四字节 |
---|---|---|---|---|
大端 | 192 | 168 | 2 | 1 |
小端 | 1 | 2 | 168 | 192 |
所以将需要比较的地址转换为网络字节序(大端模式)后,使用函数 memcmp() 就可以按字节比较两个地址大小。
int memcmp(const void *buf1, const void *buf2, unsigned int count);
比较两个IPv4地址大小
#include <stdio.h>
#include <string.h>
#include <arpa/inet.h>
int main()
{
char *str1 = "192.168.1.1";
char *str2 = "192.168.2.1";
struct in_addr ip1;
struct in_addr ip2;
int result = 0;
inet_pton(AF_INET, str1, &ip1);
inet_pton(AF_INET, str2, &ip2);
result = memcmp(&ip1, &ip2, sizeof(struct in_addr));
if (result > 0)
printf("%s is greater than %s\n", str1, str2);
else if (result < 0)
printf("%s is less than %s\n", str1, str2);
else
printf("%s is equal to %s\n", str1, str2);
return 0;
}
比较两个IPv6地址大小
#include <stdio.h>
#include <string.h>
#include <arpa/inet.h>
int main()
{
char *str1 = "2000::10";
char *str2 = "2000::2";
struct in6_addr ip1;
struct in6_addr ip2;
int result = 0;
inet_pton(AF_INET6, str1, &ip1);
inet_pton(AF_INET6, str2, &ip2);
result = memcmp(&ip1, &ip2, sizeof(struct in6_addr));
if (result > 0)
printf("%s is greater than %s\n", str1, str2);
else if (result < 0)
printf("%s is less than %s\n", str1, str2);
else
printf("%s is equal to %s\n", str1, str2);
return 0;
}