PHP安裝GeoIP擴(kuò)展根據(jù)IP獲取地理位置及計(jì)算距離的方法
根據(jù)IP獲取訪客所在國(guó)家/城市/經(jīng)緯度
安裝GeoIP擴(kuò)展:
sudo apt-get install libgeoip-dev
pecl install geoip-1.1.0
注意:Beta版要指定版本號(hào).如果是apt安裝的PHP,直接安裝php5-geoip這個(gè)包即可.
php.ini中加入:
extension=geoip.so geoip.custom_directory="/usr/share/GeoIP"
免費(fèi)下載GeoLiteCity數(shù)據(jù)庫(kù)(解壓后18MB):
http://dev.maxmind.com/geoip/legacy/install/city/
wget http://geolite.maxmind.com/download/geoip/database/GeoLiteCity.dat.gz gunzip GeoLiteCity.dat.gz sudo mkdir -v /usr/share/GeoIP sudo mv -v GeoLiteCity.dat /usr/share/GeoIP/GeoIPCity.dat
測(cè)試:
php -a
<?php print_r(geoip_record_by_name('106.37.165.80')); //回車(chē)后按Ctrl+D運(yùn)行 Array ( [continent_code] => AS [country_code] => CN [country_code3] => CHN [country_name] => China //國(guó)家 [region] => 22 [city] => Beijing //城市 [postal_code] => [latitude] => 39.928901672363 //緯度 [longitude] => 116.38829803467 //經(jīng)度 [dma_code] => 0 [area_code] => 0 )
在命令行用geoiplookup查看IP信息:
traceroute www.oschina.net
可見(jiàn)IP地址
61.145.122.155
sudo apt-get install geoip-bin geoip-database geoiplookup 61.145.122.155 -f /usr/share/GeoIP/GeoIP.dat GeoIP Country Edition: CN, China
geoip-database提供的GeoIP.dat只能精確到國(guó)家.
geoiplookup 61.145.122.155 -f /usr/share/GeoIP/GeoIPCity.dat GeoIP City Edition, Rev 1: CN, 30, Guangdong, Guangzhou, N/A, 23.116699, 113.250000, 0, 0
從maxmind官網(wǎng)下的數(shù)據(jù)庫(kù)GeoLiteCity則信息更詳細(xì).
geoiplookup 61.145.122.155 則同時(shí)顯示上述兩個(gè)數(shù)據(jù)庫(kù)的信息.
根據(jù)IP確定經(jīng)緯度與計(jì)算距離
可以用
geoip_record_by_name($_SERVER['REMOTE_ADDR'])
根據(jù)用戶IP確定經(jīng)緯度.
注意:
geoip_record_by_name()
返回的西經(jīng)和南緯是負(fù)數(shù).
5000米轉(zhuǎn)成經(jīng)緯度:
緯度 Latitude: 1 deg = 110852 m
經(jīng)度 Longitude: 1 deg = 111320*cos(lat) m
同一經(jīng)線上,相差一緯度約為 110852 米
同一緯線上,相差一經(jīng)度約為 111320*cos(lat) 米 (lat為該緯線的緯度)
<?php //以當(dāng)前用戶經(jīng)緯度為中心,查詢5000米內(nèi)的其他用戶 $y = 5000 / 110852; //緯度的范圍 $x = 5000 / (111320*cos($lat)); //經(jīng)度的范圍 $sql = ' select * from user where lat >= ($lat-$y) and lat <= ($lat+$y) and lon >= ($lon-$x) and lon <= ($lon+$x); ';
數(shù)據(jù)庫(kù)用戶表中設(shè)兩個(gè)字段,分別存儲(chǔ)用戶的經(jīng)度lat和緯度lon.
($lat-$y) <= lat <= ($lat+$y) ($lon-$x) <= lon <= ($lon+$x)
這個(gè)范圍是一個(gè)粗略的范圍,下面計(jì)算距離后把超過(guò)5公里的用戶去掉即可.
根據(jù)上面查詢出來(lái)的用戶的經(jīng)緯度,
用半正矢公式(Haversine)根據(jù)經(jīng)緯度計(jì)算兩點(diǎn)間距離:
<?php function distance($lat1, $lon1, $lat2, $lon2) { $R = 6371393; //地球平均半徑,單位米 $dlat = deg2rad($lat2-$lat1); $dlon = deg2rad($lon2-$lon1); $a = pow(sin($dlat/2), 2) + cos(deg2rad($lat1)) * cos(deg2rad($lat2)) * pow(sin($dlon/2), 2); $c = 2 * atan2(sqrt($a), sqrt(1-$a)); $d = $R * $c; return round($d); } echo distance(0, 0, -1, 0); // 111202米
然后就可以用uasort或array_multisort由近到遠(yuǎn)列出用戶了,比如有名為win,osx,lin這3個(gè)用戶:
<?php $arr = array( 'win' => array( 'dis' => 1024, 'age' => 31 ), 'osx' => array( 'dis' => 512, 'age' => 15 ), 'lin' => array( 'dis' => 512, 'age' => 25 ) ); foreach($arr as $k => $v) { $sort['dis'][$k] = $v['dis']; $sort['age'][$k] = $v['age']; } //先按距離升序排序,如果距離相同,則按年齡降序排序 array_multisort($sort['dis'], SORT_ASC, $sort['age'], SORT_DESC, $arr); echo json_encode($arr); //{"lin":{"dis":512,"age":25},"osx":{"dis":512,"age":15},"win":{"dis":1024,"age":31}}
相關(guān)文章
Zend Framework實(shí)現(xiàn)多服務(wù)器共享SESSION數(shù)據(jù)的方法
這篇文章主要介紹了Zend Framework實(shí)現(xiàn)多服務(wù)器共享SESSION數(shù)據(jù)的方法,詳細(xì)分析了SESSION數(shù)據(jù)共享的原理與實(shí)現(xiàn)技巧,需要的朋友可以參考下2016-03-03ThinkPHP簡(jiǎn)單使用memcache緩存的方法
這篇文章主要介紹了ThinkPHP簡(jiǎn)單使用memcache緩存的方法,結(jié)合實(shí)例形式分析了thinkPHP中memcache的簡(jiǎn)單配置與使用技巧,需要的朋友可以參考下2016-11-11php支付寶手機(jī)網(wǎng)頁(yè)支付類(lèi)實(shí)例
這篇文章主要介紹了php支付寶手機(jī)網(wǎng)頁(yè)支付類(lèi)實(shí)例,是基于Yii框架使用的支付寶接口類(lèi)文件,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2015-03-03yii2 數(shù)據(jù)庫(kù)讀寫(xiě)分離配置示例
本篇文章主要介紹了yii2 數(shù)據(jù)庫(kù)讀寫(xiě)分離配置示例,數(shù)據(jù)庫(kù)讀寫(xiě)分離是在網(wǎng)站遇到性能瓶頸的時(shí)候最先考慮優(yōu)化的步驟,有興趣的可以了解一下。2017-02-02完美實(shí)現(xiàn)wordpress禁止文章修訂和自動(dòng)保存的方法
這篇文章主要介紹了完美實(shí)現(xiàn)wordpress禁止文章修訂和自動(dòng)保存的方法,需要的朋友可以參考下2014-11-11PHP實(shí)現(xiàn)一個(gè)輕量級(jí)容器的方法
這篇文章主要介紹了PHP實(shí)現(xiàn)一個(gè)輕量級(jí)容器的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-01-01PHP中的empty、isset、isnull的區(qū)別與使用實(shí)例
今天小編就為大家分享一篇關(guān)于PHP中的empty、isset、isnull的區(qū)別與使用實(shí)例,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2019-03-03