php提供了ip2long與long2ip方法對ip地址處理。
1、ip2long — 將一個IPV4的字符串互聯網協議轉換成數字格式
int ip2long ( string $ip_address )
參數: ip_address 一個標準格式的地址。
返回值: 返回IP地址轉換后的數字 或 FALSE 如果 ip_address 是無效的。
2、long2ip — 將數字格式轉換成一個IPV4的字符串互聯網協議
string long2ip ( string $proper_address )
參數: proper_address 長整型的正確地址表示。
返回值: 返回互聯網地址作為字符串。
3、使用方法
1
2
3
4
5
6
|
<?php $ip = '10.1.1.1' ; $ip_long = ip2long ( $ip ); echo $ip_long .PHP_EOL; // 167837953 echo long2ip( $ip_long ); // 10.1.1.1 ?> |
4、出現負數原因及處理方法
當ip地址比較大時,ip2long會出現負數:
1
2
3
4
5
6
|
<?php $ip = '192.168.101.100' ; $ip_long = ip2long ( $ip ); echo $ip_long .PHP_EOL; // -1062705820 echo long2ip( $ip_long ); // 192.168.101.100 ?> |
原因說明:
IPv4使用無符號32位地址,因此最多有2的32次方減1(4294967295)個地址。書寫用4個小數點分開的10進制數。
記為A.B.C.D,例如:192.168.100.100。
IPv4地址每個10進制數都是無符號的字節,范圍在0~255,將IPv4地址轉為無符號數,其實就是將每個10進制數放在對應的8位上,組成一個4字節的無符號整型。192.168.100.100,192,168在高8位100,100在低8位。
C實現的例子:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
#include <stdio.h> int main( int argc, char ** argv) { unsigned int ip_long = (192 << 24) | (168 << 16) | (100 << 8) | 100; printf( "%u\n" , ip_long); printf( "%d\n" , ip_long); return 0; } fdipzone@ubuntu:~/C$ gcc -o ip2long ip2long.c fdipzone@ubuntu:~/C$ ./ip2long 3232261220 -1062706076 |
可以看到,即使ip_long聲明是無符號整型,輸出時依然需要指明%u來格式化輸出為無符號整型。
因為192大于127(二進制為01111111),192(8位)用二進制表示,最高位必然是1。導致這個4字節整型的最高位為1。
雖然ip_long定義為無符號整型,但printf方法是不理會聲明的。所以需要使用%u格式化來輸出。如果最高位是0,則使用%d即可。
另一個例子:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
ip:112.24.55.99 # include <stdio.h> int main(int argc, char** argv) { unsigned int ip_long = (112 << 24) | (24 << 16) | (55 << 8) | 99; printf( "%u\n" , ip_long); printf( "%d\n" , ip_long); return 0; } fdipzone@ubuntu:~/C$ gcc -o ip2long ip2long .c fdipzone@ubuntu:~/C$ ./ ip2long 1880635235 1880635235 |
解決方法:
輸出時用%u來格式化為無符號整型。
1
2
3
4
5
6
|
<?php $ip = '192.168.101.100' ; $ip_long = sprintf( '%u' , ip2long ( $ip )); echo $ip_long .PHP_EOL; // 3232261476 echo long2ip( $ip_long ); // 192.168.101.100 ?> |
以上這篇詳談php ip2long 出現負數的原因及解決方法就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支持服務器之家。