这里主要介绍一下 IPV4 / IPV6 在 PHP / MySQL 中如何转换。以及中间容易碰到的一些问题。

  • 首先介绍两个函数:

ip2long:将 IPV4 的字符串互联网协议转换成长整型数字

int ip2long ( string $ip_address )

long2ip:将长整型转化为字符串形式带点的互联网标准格式地址(IPV4)

string long2ip ( int $proper_address )

问题一:MySQL 中如何存储IP地址。

  • IPV4 地址长度32位,有 2^32-1 个地址。 所以 MySQL 中如果使用 int 来存储,要加 unsigned 标识。
  • int 有符号的范围是 -2^31 (-2,147,483,648) 到 2^31 - 1 (2,147,483,647) ,无符号的范围是 0 到 2^32-1(4294967295)
  • IPV6 地址长度128位。因此不能使用 int 存储,可以使用 varchar 类型存储。

问题二:ip2long 出现负数问题。

示例:

$ip_long = ip2long('192.168.8.30');
$long_ip = long2ip($ip_long);
echo $ip_long; // -1062729698
echo $long_ip; // 192.168.8.30

查看PHP手册后,发现手册上是这么介绍的:

Because PHP’s integer type is signed, and many IP addresses will result in negative integers on 32-bit architectures, you need to use the “%u” formatter of sprintf() or printf() to get the string representation of the unsigned IP address.
因为PHP的 integer 类型是有符号,并且有许多的IP地址将导致在32位系统的情况下为负数, 你需要使用 “%u” 进行转换通过 sprintf() 或printf() 得到的字符串来表示无符号的IP地址。

解决办法:

$ip_long = sprintf('%u', ip2long('192.168.8.30'));
echo $ip_long; //

接着又发现一个新问题,如果是通过 “%u” 进行转换后再调用 long2ip,会提示错误:

long2ip() expects parameter 1 to be integer,string given

接着查手册,PHP手册上是这么介绍的:

On 32-bit architectures, casting integer representations of IP addresses from string to integer is not suppossed to give correct results for numbers which exceed PHP_INT_MAX.
在 32 位架构中,从 string 转换 integer 整型形式的 ip 地址将有可能导致错误的结果,因为结果数字超出了 PHP_INT_MAX 限制。

最终解决办法,封装两个方法:

/**
* 代替 long2ip 函数
* @param $ip
* @return string
*/
function convertIpToString($ip)
{
$long = 4294967295 - ($ip - 1);
return long2ip(-$long);
} /**
* 代替 ip2long 函数
* @param $ip
* @return string
*/
function convertIpToLong($ip)
{
return sprintf("%u", ip2long($ip));
}

测试调用:

$ip_long = $this->convertIpToLong('192.168.8.30');
$long_ip = $this->convertIpToString($ip_long); echo $ip_long; //
echo $long_ip; // 192.168.8.30

问题三:MySQL 中怎么转换 IP 地址。

  • MySQL 中提供了几个函数,INET_ATON 将 IPV4 地址转换为整数。 INET_NTOA 将整数转换为 IPV4 地址。
  • 如果是 IPV6 地址也有对应的方法:INET6_ATON 和 INET6_NTOA,这两个方法需要 5.6 以上版本才能使用。

调用示例:

SELECT INET_ATON('192.168.8.30');       //
SELECT INET_NTOA('3232237598'); // 192.168.8.30

问题四:PHP 中怎么处理 IPV6 地址。

  • PHP 中没有直接提供函数实现 IPV6 地址的转换。 不过PHP手册中提供了两个方法可以实现这一需求。要运行这两个方法首先需要开启 php_gmp.dll 模块。
/**
* IPV6 地址转换为整数
* @param $ipv6
* @return string
*/
function ip2long6($ipv6)
{
$ip_n = inet_pton($ipv6);
$bits = 15; // 16 x 8 bit = 128bit
$ipv6long = '';
while ($bits >= 0) {
$bin = sprintf("%08b", (ord($ip_n[$bits])));
$ipv6long = $bin . $ipv6long;
$bits--;
}
return gmp_strval(gmp_init($ipv6long, 2), 10);
} /**
* 整数转换为 IPV6 地址
* @param $ipv6long
* @return string
*/
function long2ip6($ipv6long)
{ $bin = gmp_strval(gmp_init($ipv6long, 10), 2);
if (strlen($bin) < 128) {
$pad = 128 - strlen($bin);
for ($i = 1; $i <= $pad; $i++) {
$bin = "0" . $bin;
}
}
$bits = 0;
$ipv6 = '';
while ($bits <= 7) {
$bin_part = substr($bin, ($bits * 16), 16);
$ipv6 .= dechex(bindec($bin_part)) . ":";
$bits++;
}
// compress
return inet_ntop(inet_pton(substr($ipv6, 0, -1)));
}

测试调用:

$ip6_long = $this->ip2long6('2001:4860:a005::68');
$long_ip6 = $this->long2ip6($ip6_long);
echo $ip6_long; //
echo $long_ip6; // 2001:4860:a005::68

PHP中有关IPV4 和IPV6地址转换以及其它一些常见问题的更多相关文章

  1. (转)协议森林04 地址耗尽危机 (IPv4与IPv6地址)

    协议森林04 地址耗尽危机 (IPv4与IPv6地址) 作者:Vamei 出处:http://www.cnblogs.com/vamei 欢迎转载,也请保留这段声明.谢谢! IP地址是IP协议的重要组 ...

  2. IPv4 和 IPv6地址

    目前Internet上使用的基本都是IPv4地址,也就是说地址总共是32个比特位,也就是32位二进制数.  所以IPv4地址总的容量是 2的32次方 = 4294967296 比如 11010010 ...

  3. IP地址(IPv4)/IPv6地址的正则表达式

    原地址:http://pfeishao.blog.163.com/blog/static/18162337020112113130453/ Pv4地址正则表达式:^((25[0-5]|2[0-4]\d ...

  4. 校验IPv4和IPv6地址和URL地址

    1.校验IPV4地址: function validateIp(obj) { var ip=$(obj).val(); var re=/^(\d+)\.(\d+)\.(\d+)\.(\d+)$/;// ...

  5. 一种转换Ipv6地址的方法

    原CSDN博客不再更新维护. 本文介绍了一种将char* 类型的Ipv6地址转换成BYTE(unsigned char)或者in6_addr类型的方法. 说明:使用时需要下载Ipv6++.lib和Ip ...

  6. 在UNC(通用命名规范)路径和URL中使用IPv6地址

    转自:http://www.ipv6bbs.cn/thread-348-1-1.html   虽然微软在支持IPv6上表现得很积极,但Windows却并没有完整地支持IPv6,例如,在Windows中 ...

  7. ipv6地址在URL中的表达方式,如何在URL地址中包含ipv6地址

    摘要 本文档定义了在WWW浏览器的URL中执行的文本IPv6地址的格式.在包括Microsoft的IE,Mozilla和Lynx等几个已经被广泛安装使用的浏览器的IPv6版本中,这种格式已经被使用.并 ...

  8. URL中使用IPv4,IPv6和主机名

    在浏览器的Http请求的URL中如何使用IPv4,IPv6和主机名, 因为IPv6的地址需要加[],导致用法有点区别, 下面通过具体的例子总结一下不同情况下的用法. 1.假设有台Linux主机名配置如 ...

  9. IPV6地址中的%号什么意思

    在我配置ipv6 网络中,会发现ipv6地址后有%号,这表示什么呢? IPv6地址中的百分号是网卡interface标识.这个表示该地址仅限于标号为21的网络接口(一般指网卡或者虚拟网卡).而在其他网 ...

随机推荐

  1. 用nodejs做一个svn密码修改页面

    linux上配置好svn服务后,管理修改密码还得去手工修改passwd这个文件,略麻烦,其实网上应该有配套的web管理修改界面程序.但我想自己用nodejs写一个,因为用node不用配置复杂的服务器. ...

  2. ezdpl:完全依赖脚本和ssh的自动化部署方案

    ezdpl是easy deployment的简写,使用简单的ssh和shell脚本来部署.升级.回滚和重新配置linux服务器. 重要提示:警告:这个项目还处于测试过程中,请仔细阅读说明,并且自己承担 ...

  3. ideal环境maven自动下载项目依赖配置

    开篇序言 最近在使用ideal软件对springboot进行开发研究,中间遇到的环境和创建springboot遇到的问题真是玲琅满目,但是遇到问题就要克服这是万年不变的真理. 该文档会后续不断的补充, ...

  4. 16_Queue_利用wait()和notify()编写一个阻塞队列

    [线程间通信概念] 线程是操作系统中独立的个体,但这些个体如果不经过特殊的处理就不能成为一个整体,线程间的通信就成为整体必用方式之一.当线程存在通信指挥,线程间的交互性会更强大,在提高CPU利用率的同 ...

  5. Android 获取系统时间和网络时间

    有些时候我们的应用中只能使用网络时间,而不能使用系统的时间,这是为了避免用户关闭了使用网络时间的功能后所产生的误差. 直接上代码. 1.清单文件中网络添加权限. <!-- 访问Internet资 ...

  6. Tesorflow源代码安装方式以及错误的解决方法

    作者 修雨轩陈@cnblog 目录 Configure the installation. (安装配置) Create the pip package and install (创建pip安装包并且安 ...

  7. 《ArcGIS Runtime SDK for Android开发笔记》——(10)、ArcGIS Runtime SDK支持的空间数据类型

    1.前言 移动端的数据来源非常重要,它决定了移动端功能的实现.早期的ArcGIS Android API中,主要以接入在线的数据源为主,因此主要实现在线的地图浏览.查询和路径分析.地理处理等从操作:在 ...

  8. 在linux环境下搭建java web测试环境(非常详细!)

    一.项目必备软件及基本思路 项目必备:虚拟机:VMware Workstation (已安装linux的 CentOS6.5版本) 项目:java web项目 (必须在本地部署编译后选择项目的webR ...

  9. 常见WEB开发安全漏洞 原因分析及解决

    目 录 1 会话标识未更新 3 1.1 原因 3 1.2 解决 3 2 SQL注入 3 2.1 原因 3 2.2 解决 5 3 XSS跨站脚本编制 5 3.1 原因 5 3.2 解决 5 4 XSRF ...

  10. Java应用程序

    示例: public class HelloWorld { public static void main(String[] args) { System.out.println("Hell ...