站長(zhǎng)/程序員工具之PHP單程序文件實(shí)現(xiàn)IP歸屬地批量查詢(xún)
支持提取IP便于直接粘貼日志,去重防止重復(fù)查詢(xún)浪費(fèi)計(jì)算資源文章來(lái)源:http://www.zghlxwxcb.cn/news/detail-792728.html
按出現(xiàn)次數(shù)排序方便分析該IP行為。查詢(xún)速度極快。文章來(lái)源地址http://www.zghlxwxcb.cn/news/detail-792728.html
<?php
//整理與分享:yujianyue<15058593138@qq.com>
//注意還得放純真IP數(shù)據(jù)庫(kù) qqwry.dat(自行百度下載最新版) 和本查詢(xún)代碼同級(jí)
//核心查詢(xún)程序 class IpLocation 來(lái)自網(wǎng)絡(luò)
function tipx($str){
preg_match_all('/\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}/', $str, $match);
//$uni = array_unique($match[0]); //asort($uni);
$uni = array_count_values($match[0]); // 使用array_count_values函數(shù)統(tǒng)計(jì)數(shù)組中每個(gè)元素的數(shù)量
arsort($uni);
return $uni;//得結(jié)果,可輸出查看或調(diào)用
}
class IpLocation {
//數(shù)據(jù)文件指針
var $fp;
var $firstip;
var $lastip;
var $totalip;
function getlong() {
//unpack從二進(jìn)制字符串對(duì)數(shù)據(jù)進(jìn)行解包
//將讀取的little-endian編碼的4個(gè)字節(jié)轉(zhuǎn)化為長(zhǎng)整型數(shù),fread安全讀取二進(jìn)制文件
$result = unpack('Vlong', fread($this->fp, 4));
return $result['long'];
}
function getlong3() {
//將讀取的little-endian編碼的3個(gè)字節(jié)轉(zhuǎn)化為長(zhǎng)整型數(shù)
$result = unpack('Vlong', fread($this->fp, 3).chr(0));
return $result['long'];
}
function packip($ip) {
//pack把數(shù)據(jù)裝入一個(gè)二進(jìn)制字符串
//ip2long將IP地址轉(zhuǎn)成無(wú)符號(hào)的長(zhǎng)整型,也可以用來(lái)驗(yàn)證IP地址
return pack('N', intval(ip2long($ip)));
}
function getstring($data = "") {
$char = fread($this->fp, 1);
while (ord($char) > 0) { //ord返回字符的ASCII值,字符串按照C格式保存,以\0結(jié)束
$data .= $char;
$char = fread($this->fp, 1);
}
return $data;
}
function getarea() {
$byte = fread($this->fp, 1); // 標(biāo)志字節(jié)
switch (ord($byte)) {
case 0: // 沒(méi)有區(qū)域信息
$area = "";
break;
case 1:
case 2: // 標(biāo)志字節(jié)為1或2,表示區(qū)域信息被重定向
fseek($this->fp, $this->getlong3());
$area = $this->getstring();
break;
default: // 否則,表示區(qū)域信息沒(méi)有被重定向
$area = $this->getstring($byte);
break;
}
return $area;
}
function getlocation($ip) {
if (!$this->fp) return null; // 如果數(shù)據(jù)文件沒(méi)有被正確打開(kāi),則直接返回空
$location['ip'] = gethostbyname($ip); // 域名轉(zhuǎn)化為IP地址
$ip = $this->packip($location['ip']); // 將輸入的IP地址轉(zhuǎn)化為可比較的IP地址
// 不合法的IP地址會(huì)被轉(zhuǎn)化為255
// 對(duì)分搜索
$l = 0; // 搜索的下邊界
$u = $this->totalip; // 搜索的上邊界
$findip = $this->lastip; // 如果沒(méi)有找到就返回最后一條IP記錄(QQWry.Dat的版本信息)
while ($l <= $u) { // 當(dāng)上邊界小于下邊界時(shí),查找失敗
$i = floor(($l + $u) / 2); // 計(jì)算近似中間記錄
fseek($this->fp, $this->firstip + $i * 7);
$beginip = strrev(fread($this->fp, 4)); // 獲取中間記錄的開(kāi)始IP地址,strrev反轉(zhuǎn)字符串
// strrev函數(shù)在這里的作用是將little-endian的壓縮IP地址轉(zhuǎn)化為big-endian的格式,便于比較
//關(guān)于little-endian與big-endian 參考:http://baike.baidu.com/view/2368412.htm
if ($ip < $beginip) { // 用戶(hù)的IP小于中間記錄的開(kāi)始IP地址時(shí)
$u = $i - 1; // 將搜索的上邊界修改為中間記錄減一
}
else {
fseek($this->fp, $this->getlong3());
$endip = strrev(fread($this->fp, 4)); // 獲取中間記錄的結(jié)束IP地址
if ($ip > $endip) { // 用戶(hù)的IP大于中間記錄的結(jié)束IP地址時(shí)
$l = $i + 1; // 將搜索的下邊界修改為中間記錄加一
}
else { // 用戶(hù)的IP在中間記錄的IP范圍內(nèi)時(shí)
$findip = $this->firstip + $i * 7;
break; // 則表示找到結(jié)果,退出循環(huán)
}
}
}
fseek($this->fp, $findip);
$location['beginip'] = long2ip($this->getlong()); // 用戶(hù)IP所在范圍的開(kāi)始地址
$offset = $this->getlong3();
fseek($this->fp, $offset);
$location['endip'] = long2ip($this->getlong()); // 用戶(hù)IP所在范圍的結(jié)束地址
$byte = fread($this->fp, 1); // 標(biāo)志字節(jié)
switch (ord($byte)) {
case 1: // 標(biāo)志字節(jié)為1,表示國(guó)家和區(qū)域信息都被同時(shí)重定向
$countryOffset = $this->getlong3(); // 重定向地址
fseek($this->fp, $countryOffset);
$byte = fread($this->fp, 1); // 標(biāo)志字節(jié)
switch (ord($byte)) {
case 2: // 標(biāo)志字節(jié)為2,表示國(guó)家信息又被重定向
fseek($this->fp, $this->getlong3());
$location['country'] = $this->getstring();
fseek($this->fp, $countryOffset + 4);
$location['area'] = $this->getarea();
break;
default: // 否則,表示國(guó)家信息沒(méi)有被重定向
$location['country'] = $this->getstring($byte);
$location['area'] = $this->getarea();
break;
}
break;
case 2: // 標(biāo)志字節(jié)為2,表示國(guó)家信息被重定向
fseek($this->fp, $this->getlong3());
$location['country'] = $this->getstring();
fseek($this->fp, $offset + 8);
$location['area'] = $this->getarea();
break;
default: // 否則,表示國(guó)家信息沒(méi)有被重定向
$location['country'] = $this->getstring($byte);
$location['area'] = $this->getarea();
break;
}
if ($location['country'] == " CZNET") { // CZNET表示沒(méi)有有效信息
$location['country'] = "未知";
}
if ($location['area'] == " CZNET") {
$location['area'] = "";
}
return $location;
}
/**
* 構(gòu)造函數(shù),打開(kāi) QQWry.Dat 文件并初始化類(lèi)中的信息
*/
function __construct($filename = "./qqwry.dat") {
$this->fp = 0;
if (($this->fp = @fopen($filename, 'rb')) !== false) {
$this->firstip = $this->getlong();
$this->lastip = $this->getlong();
$this->totalip = ($this->lastip - $this->firstip) / 7;
//注冊(cè)析構(gòu)函數(shù),使其在程序執(zhí)行結(jié)束時(shí)執(zhí)行
register_shutdown_function(array(&$this, '_IpLocation'));
}
}
/**
* 析構(gòu)函數(shù),用于在頁(yè)面執(zhí)行結(jié)束后自動(dòng)關(guān)閉打開(kāi)的文件
*/
function _IpLocation() {
if ($this->fp) {
fclose($this->fp);
}
$this->fp = 0;
}
}
if($_GET["x"] == "cha"){
$tips = isset($_POST['tips']) ? $_POST['tips'] : '';
//if(!filter_var($user, FILTER_VALIDATE_EMAIL)){ exit("電子郵件格式錯(cuò)誤");}
$txts = str_replace(array("\r\n","\r","\n","\t",","," "),"|",$tips);
$lisa = tipx($txts); //explode("|",$txts);
echo "<h2><strong>以下信息僅供參考</strong>(提取IP/已去重/按出現(xiàn)次數(shù)降序排序):</h2>\r\n";
echo "<table cellspacing=\"0\" class=\"table\" cellpadding=\"0\">\r\n";
echo "<tr class='tt'><td width='99'>IP</td><td width='60'>出現(xiàn)次數(shù)</td><td width='99'>IP/24</td><td>歸屬地</td></tr>\r\n";
$iplocation = new IpLocation(); $ipx= array();
foreach($lisa as $zz => $ges){
$location = $iplocation->getlocation($zz);
$zi = explode(".",$zz); $zip = $zi[0].".".$zi[1].".".$zi[2].".0/24";
$diqu = mb_convert_encoding($location["country"]."_".$location["area"], "utf-8", "gbk");
echo "<tr><td>$zz</td><td>$ges</td><td>$zip</td><td>$diqu</td></tr>\r\n";
}
echo "</table>\r\n";
exit();
}
?>
<!DOCTYPE html>
<html>
<head>
<title>批量查IP歸屬地</title>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width,minimum-scale=1.0,maximum-scale=1.0">
<meta name="apple-mobile-web-app-capable" content="yes" />
<script type="text/javascript">
console.log("問(wèn)題反饋電話:","15058593138");
console.log("問(wèn)題反饋郵件:","admin@12391.net");
function $(objId){
return document.getElementById(objId);
}
function loadcha(xid) {
var xmlhttp;
var Stxt= "nums=aa";
Stxt+="&tips="+ encodeURIComponent($("tips").value);
//$("tips").innerHTML = "正在加載...";
if (window.XMLHttpRequest) {
xmlhttp = new XMLHttpRequest();
} else {
xmlhttp = new ActiveXObject("Microsoft.XMLHTTP");
}
xmlhttp.onreadystatechange = function() {
if (xmlhttp.readyState == 4 && xmlhttp.status == 200) {
var btxt = xmlhttp.response;
if(btxt == "err01"){ $("tipx").innerHTML = "!"; return false;}
$('tipx').innerHTML = xmlhttp.response;
}
}
xmlhttp.open("POST", "?x=cha&tt="+Math.random(), true);
xmlhttp.setRequestHeader("Content-type", "application/x-www-form-urlencoded");
xmlhttp.send(Stxt);
}
</script>
<style>
div,#tipx{display:block;width:99.7%;border:0;margin-top:5px;}
textarea{display:block;width:99.7%;border:1px solid #ccc;height:160px;}
table{margin:20px auto;border-left:1px solid #a2c6d3;border-top:3px solid #0180CF;font-size:12px;width:99.7%;}
table td{border-right:1px solid #a2c6d3;border-bottom:1px solid #a2c6d3;padding:2px;word-wrap:break-word;word-break:break-all;}
td{min-width:30px;max-width:490px;}
.tt{background-color: #f2f2f2;}
#submit{ height:35px;}
</style>
</head>
<body>
<form class="form" id="form" method="POST" act="?act=cha" >
<h3>批量查詢(xún)IP歸屬地</h3>
<p>自動(dòng)提文本中IP地址去重和出現(xiàn)次數(shù)降序并查詢(xún)歸屬地!</p>
<textarea id="tips">
3.224.220.24
52.70.240.24
202.63.172.24
</textarea>
<input type="button" id="submit" value="提交查詢(xún)" onclick="loadcha('xid')">
<div id="tipx"></div>
</form>
</body>
</html>
到了這里,關(guān)于站長(zhǎng)工具之PHP單文件實(shí)現(xiàn)IP歸屬地批量查詢(xún)的文章就介紹完了。如果您還想了解更多內(nèi)容,請(qǐng)?jiān)谟疑辖撬阉鱐OY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!