<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <title>PHP中文网</title> <meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> </head> <body> <?php /* * 作者:XXXX */ //将IP转换为数字 function ipton($ip) { $ip_arr=explode('.',$ip);//分隔ip段 foreach ($ip_arr as $value) { $iphex=dechex($value);//将每段ip转换成16进制 if(strlen($iphex)<2)//255的16进制表示是ff,所以每段ip的16进制长度不会超过2 { $iphex='0'.$iphex;//如果转换后的16进制数长度小于2,在其前面加一个0 //没有长度为2,且第一位是0的16进制表示,这是为了在将数字转换成ip时,好处理 } $ipstr.=$iphex;//将四段IP的16进制数连接起来,得到一个16进制字符串,长度为8 } return hexdec($ipstr);//将16进制字符串转换成10进制,得到ip的数字表示 } //将数字转换为IP,进行上面函数的逆向过程 function ntoip($n) { $iphex=dechex($n);//将10进制数字转换成16进制 $len=strlen($iphex);//得到16进制字符串的长度 if(strlen($iphex)<8) { $iphex='0'.$iphex;//如果长度小于8,在最前面加0 $len=strlen($iphex); //重新得到16进制字符串的长度 } //这是因为ipton函数得到的16进制字符串,如果第一位为0,在转换成数字后,是不会显示的 //所以,如果长度小于8,肯定要把第一位的0加上去 //为什么一定是第一位的0呢,因为在ipton函数中,后面各段加的'0'都在中间,转换成数字后,不会消失 for($i=0,$j=0;$j<$len;$i=$i+1,$j=$j+2) {//循环截取16进制字符串,每次截取2个长度 $ippart=substr($iphex,$j,2);//得到每段IP所对应的16进制数 $fipart=substr($ippart,0,1);//截取16进制数的第一位 if($fipart=='0') {//如果第一位为0,说明原数只有1位 $ippart=substr($ippart,1,1);//将0截取掉 } $ip[]=hexdec($ippart);//将每段16进制数转换成对应的10进制数,即IP各段的值 } $ip = array_reverse($ip); return implode('.', $ip);//连接各段,返回原IP值 } echo ipton('119.255.31.226'); echo '<br>'; $num='379374783'; echo strlen($num).'<br/>'; echo ntoip($num).'<br/>'; echo 'trueipnum:'.ip2long('119.255.31.226').'<br/>'; echo 'trueip:'.long2ip('3793747831'); ?> </body> </html>
mysql には、ip アドレスを数値に変換する inet_aton 関数と、数値を ip に変換する inet_ntoa 関数が付属しています。
SELECT INET_ATON('10.122.22.1')
上記の変換関数は、偶然にも mysql システム関数と同じ変換効果を実現します。 PHP には、IP と数値の変換を実現する ip2long 関数と long2ip2 関数が用意されていることがわかりました。ウェブマスターは無駄な仕事をしましたが、論理的思考を働かせる必要があります~~
これらは 2 つの実装方法で、1 つはプログラムで実装する方法です。もう 1 つは、SQL ステートメントに直接実装する方法です。非常に便利で優れています。将来忘れないように保存してブログに投稿しました。
IP データをデータベース (MySQL) に保存する場合、ip2long 関数を使用して整数を生成し、それを int(11) 型フィールドに格納することに慣れていますが、異なるシステム プラットフォームでは、ip2long 関数が取得します。値が異なるため、データベースからデータを読み取り、long2ip を使用して IP を取得するときにエラーが発生する可能性があります。
int (11) 型 (範囲 - 2147483648 - 2147483647) を使用します。 ip2long を使用して IP アドレスを処理した結果を保存します。たとえば、ip が「202.105.77.179」の場合、32 ビット マシンで得られる結果は -899068493 ですが、64 ビット マシンでは 3395898803 になります。データベースが int(11) の範囲を超えているため、64 ビット マシン上の結果は int(11) の最大値 2147483647 として保存されます。間違った結果が得られ、「127.255.255.255」この IP アドレスです。
解決策は多数あります。たとえば、mysql 関数 INET_ATON および INET_NTOA を使用して IP アドレスを処理するか、IP アドレスを保存するフィールドを変更することができます。 bigint 型なので、64 ビット マシンに保存できます。これは 3395898803 ですが、long2ip 関数を使用すると正しい結果が得られます。