根据掩码计算ip范围值。实际可用的范围还要-2(全0和全1)
- $ipstr = "192.168.178.123/24"; 
- list($ipstr, $maskstr) = explode('/', $ipstr); 
- $base = ip2long('255.255.255.255'); 
- $mask = pow(2,32-intval($maskstr))-1;//mask=0.0.0.255(int) 
- $smask = $mask ^ $base;//smask=255.255.255.0(int) 
- echo long2ip($min)."\n";//192.168.178.0 
- echo long2ip($max)."\n";//192.168.178.255 
来自 https://blog.csdn.net/pupur99/article/details/17244075 
php 怎样将有范围的ip转化为整型范围
php中将IP转换成整型的函数ip2long()容易出现问题,在IP比较大的情况下,会变成负数。
如下:
- <?php
 
 
- $ip = "192.168.1.2";
 
 
- $ip_n = ip2long($ip);
 
 
- echo $ip_n;      //得到 -1062731518
 
 
- ?> 
由于IP转换成的整型值太大超出了整型的范围,所以变成负数。需写成$ip_n = bindec(decbin(ip2long($ip)));这样便可得到无符号的整型数,如下:
- <?php
 
 
- $ip = "192.168.1.2";
 
 
- $ip_n = bindec(decbin(ip2long($ip)));
 
 
- echo $ip_n;      //得到 3232235778
 
 
- ?> 
PS: 另一个方法:
- $ip_long = = sprintf("%u", ip2long($ip)); 
如何将四个字段以点分开的IP网络址协议地址转换成整数呢?PHP里有这么一个函数ip2long.比如:
- <?php
 
 
- echo ip2long("10.2.1.3");
 
 
- ?> 
我们将得到
167903491
这是如何计算的,目前我知道有两个算法。其一:
- <?php
 
 
- function ip2int($ip){
 
 
-     //我们先把ip分为四段,$ip1,$ip2,$ip3,$ip4
 
 
-     list($ip1,$ip2,$ip3,$ip4)=explode(".",$ip);
 
 
-     //然后第一段乘以256的三次方,第二段乘以256的平方,第三段乘以256
 
 
-     //这即是我们得到的值
 
 
-     return $ip1*pow(256,3)+$ip2*pow(256,2)+$ip3*256+$ip4;
 
 
- }
 
 
- ?> 
其二,用位运算:
- <?php
 
 
- function ip2int($ip){
 
 
-     list($ip1,$ip2,$ip3,$ip4)=explode(".",$ip);
 
 
-     return ($ip1<<24)|($ip2<<16)|($ip3<<8)|($ip4);
 
 
- }
 
 
- ?> 
我们会发现,有些ip转化成整数后,是负的,这是因为得到的结果是有符号整型,最大值是2147483647.要把它转化为无符号的,可以用 sprintf("%u",ip2long($ip);
就能转换为正整数。而且得到的结果用long2ip也可以正常转换回原来的ip地址。也可以用ip2long来验证一个ip是否是有效的,如:
- <?php
 
 
- function chk_ip($ip){
 
 
-     if(ip2long($ip)=="-1") {
 
 
-        return false;
 
 
-     }
 
 
-     return true;
 
 
- }
 
 
- //应用
 
 
- var_export(chk_ip("10.111.149.42"));
 
 
- var_export(chk_ip("10.111.256.42"));
 
 
- ?> 
将输出true和false
把ip数据保存在数据库(MySQL)中时候,我们习惯用ip2long函数生成整型,然后存放在一个int(11)类型的字段中,但是,在不同的系统平台上,ip2long函数得到的值是不同的,因此可能造成在从数据库中读出数据,用long2ip得到ip的时候产生错误,说一下我们碰到的情况:
我们用一个int(11)类型(范围-2147483648 - 2147483647)来保存把一个ip地址用ip2long处理得到的结果,例如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函数仍能得到正确的结果.
来自  https://www.bbsmax.com/A/ke5j2XVj5r/ 
php 掩码_php如何计算IP地址掩码
本文主要介绍了php进行ip地址掩码运算处理的方法,涉及php字符串及网络操作相关技巧,需要的朋友可以参考下。希望对大家有所帮助。
在ip分段范围当中里面
具体如下:
ip解析:function ip_parse($ip_str) {
                $mark_len = 32;
                
                if (strpos($ip_str, "/") > 0) {
                
                list($ip_str, $mark_len) = explode("/", $ip_str);
                
                }
                
                $ip = ip2long($ip_str);
                
                $mark = 0xFFFFFFFF << (32 - $mark_len) & 0xFFFFFFFF;
                
                $ip_start = $ip & $mark;
                
                $ip_end = $ip | (~$mark) & 0xFFFFFFFF;
                
                return array($ip, $mark, $ip_start, $ip_end);
                
}
//演示:
list($ip, $mark, $ip_start, $ip_end) = ip_parse("192.168.1.12/24");
echo "IP地址 : ", long2ip($ip), "\n";
echo "子网掩码: ", long2ip($mark), "\n";
echo "IP段开始: ", long2ip($ip_start), "\n";
echo "IP段结束: ", long2ip($ip_end), "\n";
结果:IP地址 : 192.168.1.12
子网掩码: 255.255.255.0
IP段开始: 192.168.1.0
IP段结束: 192.168.1.255
ip是否在ip分段中:function ip_in($ip, $ip_str) {
            $mark_len = 32;
            
            if (strpos($ip_str, "/") > 0) {
            
                        list($ip_str, $mark_len) = explode("/", $ip_str);
            
            }
            
            $right_len = 32 - $mark_len;
            
            return ip2long($ip) >> $right_len == ip2long($ip_str) >> $right_len;
            
}
//演示:
var_dump(ip_in("192.168.1.1", "192.168.1.0/24"));
来自  https://blog.csdn.net/weixin_28950415/article/details/115093314 
    为了检测客户端ip是否位于指定的网络里(如防火墙过滤有时候需要用到这个技术),有如下方法:
1、第一种
| 1 2 3 4 5 6 7 8 9 10 11 |     publicfunction netMatch($client_ip, $server_ip, $mask)
 {
     $mask1 = 32 - $mask;
             return((ip2long($client_ip) >> $mask1) == (ip2long($server_ip) >> $mask1));
 }
     $client_ip = '192.168.1.15';
     $server_ip = '192.168.1.1';
     $mask = 24;
     $res = netMatch( $dev_ip, $server_ip , $mask );
 | 
  
2、第二种
    1、客户端ip,假设符号是D
    2、该网络下的ip地址段,假设符号是N
    3、该网络下的子网掩码,假设符号是M
    那么,如果客户端ip处于该网络下,应该会满足:
    N==(D&M)
     文字解释就是,用客户端ip和子网掩码进行与操作,所得结果应该与该网络下的ip段相等。不过注意一个问题,上面的公式,是用二进制进行计算的,并不是我们常见的点分十进制的ip地址。也就是说,我们获得客户端ip后,应该把它转换为32位二进制数字再进行计算。
    代码如下:
function ip_binay($ip)
{
        $ip_array = explode(".",$ip);
        $t = '';
        foreach($ip_array as $v)
        {
                $tmp = decbin($v);
                $tmp_len = strlen(decbin($v));
                if($tmp_len < 8)
                        $t .= str_repeat( "0",(8 - $tmp_len) ).$tmp;
                else
                        $t .= $tmp;
        }
        return $t;
}
$server_ip = "0.0.0.0"; //地址段
$mask_ip = "0.0.0.0"; //子网掩码
 
$client_ip='192.168.1.11';//客户端ip
 
$D=ip_binay($client_ip);
$is_ip_in=0;
$M = ip_binay($mask_ip);
$temp = $D & $M;
$N = ip_binay( $server_ip );
if( $temp == $N ){
    $is_ip_in=1;
}
 
if($is_ip_in){
    echo "你的ip处于该网络下!";
}else{
    echo 'NO';
} 
来自 https://www.cnblogs.com/rnckty/p/4422210.html