IPv6:IP地址鉴权(比较两个IP地址是否相等)
IPv4地址长度为32bit(4B),只有点分十进制一种表示形式。
IPv4地址的点分十进制形式,最长为:
ddd.ddd.ddd.ddd
共计15字节,用16字节来存储即可。
比较两个IPv4的地址是否相等,可以通过简单的字符串比较。
IPv6地址长度为128bit(32B),使用“冒分十六进制表示法” 和/或 “0位压缩表示法” 和/或 “内嵌IPv4地址表示法”。
此时,一个实际相同的IPv6的地址,可能有很多种不同的表示形式。
我们要判断两个IPv6地址是否相同(忽略不同形式的差异),就需要一种“统一的处理手段”,将不同形式的IPv6地址,映射为唯一的确切形式,然后进行比较判断。
这种“统一的映射手段”,可以是inet_pton:
NAME
inet_pton - convert IPv4 and IPv6 addresses from text to binary form
SYNOPSIS
#include <arpa/inet.h>
int inet_pton(int af, const char *src, void *dst);
DESCRIPTION
This function converts the character string src into a network address structure in the af address family, then copies the network address structure to dst.
The af argument must be either AF_INET or AF_INET6.
RETURN VALUE
inet_pton() returns 1 on success (network address was successfully converted).
0 is returned if src does not contain a character string representing a valid network address in the specified address family.
If af does not contain a valid address family, -1 is returned and errno is set to EAFNOSUPPORT.
数据其实有很多表现形式,IPv6的字符串表现形式,可以是缩写、全写等,但是将其转换为binary形式,就能将其唯一确定。
inet_pton就是将一个IPv6字符串表现形式转换为唯一的binary形式,即使IPv6字符串表现形式可能出现全写、缩写等等,但是其binary一定是唯一的。(换句话说,binary数据,在IPv6字符串表示形式规则中可以呈现不同的字符串形式)
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
int main()
{
struct in6_addr addr1;
struct in6_addr addr2;
const char *ip1 =
"2001:0DB8:0000:0023:0008:0800:200C:417A";
// "::1";
const char *ip2 =
"2001:DB8:0:23:8:800:200C:417A";
// "0000:0000:0000:0000:0000:0000:0000:0001";
int res1 = inet_pton(AF_INET6, ip1, &addr1);
if (res1 == 0)
{
printf("invalid network address!!!\n");
exit(1);
}
else if (res1 < 0)
{
printf("inet_pton error=%d(%s)!!!\n", errno, strerror(errno));
exit(1);
}
int res2 = inet_pton(AF_INET6, ip2, &addr2);
if (res2 == 0)
{
printf("invalid network address!!!\n");
exit(1);
}
else if (res2 < 0)
{
printf("inet_pton error=%d(%s)!!!\n", errno, strerror(errno));
exit(1);
}
if (memcmp(&addr1, &addr2, sizeof(struct in6_addr)) == 0)
{
printf("addr1 equal addr2!!!\n");
}
else
{
printf("addr1 not equal addr2!!!\n");
}
return 0;
}
编译 && 运行:
[jiang@localhost ~]$ gcc -o main main.c
[jiang@localhost ~]$ ./main
addr1 equal addr2!!!
上面代码中,有两个IPv6地址,我们现将其转换为对应的binary数据,然后用memcmp来逐位进行比较。
如果两个binary数据完全相同,说明两个IPv6地址指代同一个地址,是等价的。
同理可用此方法比较IPv4地址,注意binary用struct in_addr(32bit)而非struct in6_addr(128bit)。