相關(guān)關(guān)鍵詞
關(guān)于我們
最新文章
- PHP中opcode緩存簡單用法分析
- thinkPHP控制器變量在模板中的顯示方法示例
- PHP move_uploaded_file() 函數(shù)(將上傳的文件移動(dòng)到新位置)
- dirname(__FILE__)的含義和應(yīng)用說明
- thinkPHP5框架實(shí)現(xiàn)分頁查詢功能的方法示例
- PHP中單雙號(hào)與變量
- PHP獲得當(dāng)日零點(diǎn)時(shí)間戳的方法分析
- Laravel ORM對Model::find方法進(jìn)行緩存示例詳解
- PHP讀寫文件高并發(fā)處理操作實(shí)例詳解
- 【CLI】利用Curl下載文件實(shí)時(shí)進(jìn)度條顯示的實(shí)現(xiàn)
PHP獲取IP地址所在地信息的實(shí)例(使用純真IP數(shù)據(jù)庫qqwry.dat)
如下所示:
<?php $data = '254.254.254.254'; echo ip2long($data); function getIP() { if(getenv("HTTP_CLIENT_IP") && strcasecmp(getenv("HTTP_CLIENT_IP"), "unknown")) $ip = getenv("HTTP_CLIENT_IP"); elseif(getenv("HTTP_X_FORWARDED_FOR") && strcasecmp(getenv("HTTP_X_FORWARDED_FOR"), "unknown")) $ip = getenv("HTTP_X_FORWARDED_FOR"); elseif (getenv("REMOTE_ADDR") && strcasecmp(getenv("REMOTE_ADDR"), "unknown")) $ip = getenv("REMOTE_ADDR"); elseif (isset($_SERVER['REMOTE_ADDR']) && $_SERVER['REMOTE_ADDR'] && strcasecmp($_SERVER['REMOTE_ADDR'], "unknown")) $ip = $_SERVER['REMOTE_ADDR']; else $ip = "0.0.0.0"; return $ip; } class IpLocation { //數(shù)據(jù)文件指針 var $fp; var $firstip; var $lastip; var $totalip; function getlong() { //unpack從二進(jìn)制字符串對數(shù)據(jù)進(jìn)行解包 //將讀取的little-endian編碼的4個(gè)字節(jié)轉(zhuǎn)化為長整型數(shù),fread安全讀取二進(jìn)制文件 $result = unpack('Vlong', fread($this->fp, 4)); return $result['long']; } function getlong3() { //將讀取的little-endian編碼的3個(gè)字節(jié)轉(zhuǎn)化為長整型數(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)成無符號(hào)的長整型,也可以用來驗(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: // 沒有區(qū)域信息 $area = ""; break; case 1: case 2: // 標(biāo)志字節(jié)為1或2,表示區(qū)域信息被重定向 fseek($this->fp, $this->getlong3()); $area = $this->getstring(); break; default: // 否則,表示區(qū)域信息沒有被重定向 $area = $this->getstring($byte); break; } return $area; } function getlocation($ip) { if (!$this->fp) return null; // 如果數(shù)據(jù)文件沒有被正確打開,則直接返回空 $location['ip'] = gethostbyname($ip); // 域名轉(zhuǎn)化為IP地址 $ip = $this->packip($location['ip']); // 將輸入的IP地址轉(zhuǎn)化為可比較的IP地址 // 不合法的IP地址會(huì)被轉(zhuǎn)化為255 // 對分搜索 $l = 0; // 搜索的下邊界 $u = $this->totalip; // 搜索的上邊界 $findip = $this->lastip; // 如果沒有找到就返回最后一條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)); // 獲取中間記錄的開始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) { // 用戶的IP小于中間記錄的開始IP地址時(shí) $u = $i - 1; // 將搜索的上邊界修改為中間記錄減一 } else { fseek($this->fp, $this->getlong3()); $endip = strrev(fread($this->fp, 4)); // 獲取中間記錄的結(jié)束IP地址 if ($ip > $endip) { // 用戶的IP大于中間記錄的結(jié)束IP地址時(shí) $l = $i + 1; // 將搜索的下邊界修改為中間記錄加一 } else { // 用戶的IP在中間記錄的IP范圍內(nèi)時(shí) $findip = $this->firstip + $i * 7; break; // 則表示找到結(jié)果,退出循環(huán) } } } fseek($this->fp, $findip); $location['beginip'] = long2ip($this->getlong()); // 用戶IP所在范圍的開始地址 $offset = $this->getlong3(); fseek($this->fp, $offset); $location['endip'] = long2ip($this->getlong()); // 用戶IP所在范圍的結(jié)束地址 $byte = fread($this->fp, 1); // 標(biāo)志字節(jié) switch (ord($byte)) { case 1: // 標(biāo)志字節(jié)為1,表示國家和區(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,表示國家信息又被重定向 fseek($this->fp, $this->getlong3()); $location['country'] = $this->getstring(); fseek($this->fp, $countryOffset + 4); $location['area'] = $this->getarea(); break; default: // 否則,表示國家信息沒有被重定向 $location['country'] = $this->getstring($byte); $location['area'] = $this->getarea(); break; } break; case 2: // 標(biāo)志字節(jié)為2,表示國家信息被重定向 fseek($this->fp, $this->getlong3()); $location['country'] = $this->getstring(); fseek($this->fp, $offset + 8); $location['area'] = $this->getarea(); break; default: // 否則,表示國家信息沒有被重定向 $location['country'] = $this->getstring($byte); $location['area'] = $this->getarea(); break; } if ($location['country'] == " CZNET") { // CZNET表示沒有有效信息 $location['country'] = "未知"; } if ($location['area'] == " CZNET") { $location['area'] = ""; } return $location; } /** * 構(gòu)造函數(shù),打開 QQWry.Dat 文件并初始化類中的信息 */ function IpLocation($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; //注冊析構(gòu)函數(shù),使其在程序執(zhí)行結(jié)束時(shí)執(zhí)行 register_shutdown_function(array(&$this, '_IpLocation')); } } /** * 析構(gòu)函數(shù),用于在頁面執(zhí)行結(jié)束后自動(dòng)關(guān)閉打開的文件 */ function _IpLocation() { if ($this->fp) { fclose($this->fp); } $this->fp = 0; } } header("content-Type: text/html; charset=gbk"); $ipOrDomain='110.0.0.0'; //$ipOrDomain='www.baidu.com'; $iplocation = new IpLocation(); $location = $iplocation->getlocation($ipOrDomain); $address=mb_convert_encoding($location['country'].$location['area'], "gbk", "gbk"); echo $address; ?>