2019独角兽企业重金招聘Python工程师标准>>>
1、首先我们应该先明白为什么要这样存储ip而用varchar直接存储IP呢?
其实做任何程序设计都要在功能实现的基础上最大限度的优化性能。而数据库设计是程序设计中不可忽略的一个重要部分,所以巧存IP地址可以一定程度获得很大提升。
利用函数算法处理:
在MySQL中没有直接提供IP类型字段,但如果有两个函数可以把IP与最大长度为10位数字类型互转,所以使用int类型存储IP比varchar类型存储IP地址性能要提升很多,减少不少空间。因为varchar是可变长形,需要多余的一个字节存储长度。另外int型在逻辑运算上要比varchar速度快。
接下来我们找一个ip地址分析一下它应int存储的过程:
先来看IP的结构, 127.0.0.1
学计算机的对2的n次方比较敏感,一看就猜到是由 4个8位的二进制数字组成, 对, 你猜的没错,就是这样!也就是说IP转换成二进制可以这样:
select concat(bin(127),lpad(bin(0),8,0),lpad(bin(0),8,0),lpad(bin(1),8,0));
把每一位的前边补0, 然后拼接起来,得到这样的结果: 1111111000000000000000000000001
系统的INET_ATON()函数也是这么算的,不信可以对比一下结果:
select bin(inet_aton('127.0.0.1'));
得到的结果也是这样: 1111111000000000000000000000001
所以得出结论: INET_ATON() 就是把IP的每一段转为二进制拼接起来,然后将这个32位的二进制数字转为10进制
示例:
select INET_ATON('127.0.0.1'); -----------------转为数字
select INET_NTOA(2130706433); ------------------转为IP
为什么这么存储呢?
有人说节省空间, 也对, 不过比较蛋疼, 现在的硬盘那么便宜在乎这点空间?更为合理的解释是, 方便查询, 比如要查某一网段的所有IP就可以 :
select * from table where ip between inet_aton('192.168.0.1') and inet_aton('192.168.0.255');
到这里我们就已经把如何用int类型代替varchar类型存储ip的过程分析完了。上面的分析中我们用到了两个函数INET_ATON()和INET_NTOA(),下面我们用具体的实例来说一下这两个函数的具体用法。
IP的表字段可以设置为INT(10)就好,如果IP获取不到可以直接存0代表获取不到IP的意思。
2、实例分析
将IP以整型的方式存储到数据库中
在实现某些功能时有时会用到用户的IP地址,因为IP地址有四段,在保存到MySQL数据库中的时候可能会使用varchar(15)或者char(15),其实还有更好的方式那就是将IP地址转换成整型,这样占用空间更小,查询速度快,同时还方便比较。在数据库设计时要用无符号INT UNSIGNED,否则长度不够。下面给出主要PHP代码:
$ip = '这里是获取到的IP';
echo $intip = sprintf('%u',ip2long($ip)); //转换为无符号整型
echo long2ip($intip);//将整型转换为ip
?>
这样保存到数据库之后也很容易比较,例如类似between之类的语句,在sql语句中也可以通过函数实现转换:
将IP地址转换为整型可以使用
select inet_aton(ip) from users;
将整型转换为IP地址可以使用
select inet_ntoa(intip) from users;