php查询ip所在地的方法
本文实例讲述了php查询ip所在地的方法。分享给大家供大家参考。
具体实现方法如下:
<?php
/**
*@date        2010.12.21
注:文件头[第一条索引的偏移量(4byte)]+[最后一条索引的偏移地址(4byte)]    8字节
记录区[结束ip(4byte)]+[地区1]+[地区2]                               4字节+不定长
索引区[开始ip(4byte)]+[指向记录区的偏移地址(3byte)]                  7字节
*/
classiplocation{
var$fp;
var$firstip; //第一条ip索引的偏移地址
var$lastip;  //最后一条ip索引的偏移地址
var$totalip; //总ip数
/*
|----------------------------------------------------------------------------
|构造函数,初始化一些变量
|----------------------------------------------------------------------------
|
*/
functioniplocation($datfile="qqwry.dat"){
$this->fp=fopen($datfile,'rb')ordie("qqwry.dat不存在,请去网上<ahref='https://www.nhooo.com/softs/10529.html'>下载纯真ip数据库</a>,'qqwry.dat'放到当前目录下");  //二制方式打开
$this->firstip=$this->get4b();//第一条ip索引的绝对偏移地址
$this->lastip=$this->get4b(); //最后一条ip索引的绝对偏移地址
$this->totalip=($this->lastip-$this->firstip)/7;//ip总数索引区是定长的7个字节,在此要除以7,
register_shutdown_function(array($this,"closefp")); //为了兼容php5以下版本,本类没有用析构函数,自动关闭ip库.
}
/*
|----------------------------------------------------------------------------
|关闭ip库
|----------------------------------------------------------------------------
|
*/
functionclosefp(){
fclose($this->fp);
}
/*
|----------------------------------------------------------------------------
|读取4个字节并将解压成long的长模式
|----------------------------------------------------------------------------
|
*/
functionget4b(){
$str=unpack("v",fread($this->fp,4));
return$str[1];
}
/*
|----------------------------------------------------------------------------
|读取重定向了的偏移地址
|----------------------------------------------------------------------------
|
*/
functiongetoffset(){
$str=unpack("v",fread($this->fp,3).chr(0));
return$str[1];
}
/*
|----------------------------------------------------------------------------
|读取ip的详细地址信息
|----------------------------------------------------------------------------
|
*/
functiongetstr(){
$split=fread($this->fp,1);
while(ord($split)!=0){
$str.=$split;
$split=fread($this->fp,1);
}
return$str;
}
/*
|----------------------------------------------------------------------------
|将ip通过ip2long转成ipv4的互联网地址,再将他压缩成big-endian字节序,用来和索引区内的ip地址做比较
|----------------------------------------------------------------------------
|
*/
functioniptoint($ip){
returnpack("n",intval(ip2long($ip)));
}
/*
|----------------------------------------------------------------------------
|获取地址信息
|----------------------------------------------------------------------------
|
*/
functionreadaddress(){
$now_offset=ftell($this->fp);//得到当前的指针位址
$flag=$this->getflag();
switch(ord($flag)){
case0:
$address="";
break;
case1:
case2:
fseek($this->fp,$this->getoffset());
$address=$this->getstr();
break;
default:
fseek($this->fp,$now_offset);
$address=$this->getstr();
break;
}
return$address;
}
/*
|----------------------------------------------------------------------------
|获取标志1或2  用来确定地址是否重定向了
|----------------------------------------------------------------------------
|
*/
functiongetflag(){
returnfread($this->fp,1);
}
/*
|----------------------------------------------------------------------------
|用二分查找法在索引区内搜索ip
|----------------------------------------------------------------------------
|
*/
functionsearchip($ip){
$ip=gethostbyname($ip);    //将域名转成ip
$ip_offset["ip"]=$ip;
$ip=$this->iptoint($ip);   //将ip转换成长整型
$firstip=0;                //搜索的上边界
$lastip=$this->totalip;    //搜索的下边界
$ipoffset=$this->lastip;   //初始化为最后一条ip地址的偏移地址
while($firstip<=$lastip){
$i=floor(($firstip+$lastip)/2);         //计算近似中间记录floor函数记算给定浮点数小的最大整数,说白了就是四舍五也舍
fseek($this->fp,$this->firstip+$i*7);   //定位指针到中间记录
$startip=strrev(fread($this->fp,4));        //读取当前索引区内的开始ip地址,并将其little-endian的字节序转换成big-endian的字节序
if($ip<$startip){
$lastip=$i-1;
}
else{
fseek($this->fp,$this->getoffset());
$endip=strrev(fread($this->fp,4));
if($ip>$endip){
$firstip=$i+1;
}
else{
$ip_offset["offset"]=$this->firstip+$i*7;
break;
}
}
}
return$ip_offset;
}
/*
|----------------------------------------------------------------------------
|获取ip地址详细信息
|----------------------------------------------------------------------------
|
*/
functiongetaddress($ip){
$ip_offset=$this->searchip($ip); //获取ip在索引区内的绝对编移地址
$ipoffset=$ip_offset["offset"];
$address["ip"]=$ip_offset["ip"];
fseek($this->fp,$ipoffset);     //定位到索引区
$address["startip"]=long2ip($this->get4b());//索引区内的开始ip地址
$address_offset=$this->getoffset();           //获取索引区内ip在ip记录区内的偏移地址
fseek($this->fp,$address_offset);           //定位到记录区内
$address["endip"]=long2ip($this->get4b());  //记录区内的结束ip地址
$flag=$this->getflag();                     //读取标志字节
switch(ord($flag)){
case1: //地区1地区2都重定向
$address_offset=$this->getoffset();  //读取重定向地址
fseek($this->fp,$address_offset);    //定位指针到重定向的地址
$flag=$this->getflag();              //读取标志字节
switch(ord($flag)){
case2: //地区1又一次重定向,
fseek($this->fp,$this->getoffset());
$address["area1"]=$this->getstr();
fseek($this->fp,$address_offset+4);     //跳4个字节
$address["area2"]=$this->readaddress(); //地区2有可能重定向,有可能没有
break;
default://地区1,地区2都没有重定向
fseek($this->fp,$address_offset);       //定位指针到重定向的地址
$address["area1"]=$this->getstr();
$address["area2"]=$this->readaddress();
break;
}
break;
case2://地区1重定向地区2没有重定向
$address1_offset=$this->getoffset();  //读取重定向地址
fseek($this->fp,$address1_offset);  
$address["area1"]=$this->getstr();
fseek($this->fp,$address_offset+8);
$address["area2"]=$this->readaddress();
break;
default://地区1地区2都没有重定向
fseek($this->fp,$address_offset+4);
$address["area1"]=$this->getstr();
$address["area2"]=$this->readaddress();
break;
}
//*过滤一些无用数据
if(strpos($address["area1"],"cz88.net")!=false){
$address["area1"]="未知";
}
if(strpos($address["area2"],"cz88.net")!=false){
$address["area2"]="";
}
return$address;
}
}
 
/*用法如下:*/
$ip=newiplocation("qqwry.dat");
$address=$ip->getaddress("61.129.51.27");
//$address=$ip->getaddress(www.nhooo.com);
echo'<pre>';
print_r($address);
?>
希望本文所述对大家的PHP程序设计有所帮助。
