搜索附近的人PHP實(shí)現(xiàn)代碼
本文實(shí)例為大家分享了PHP實(shí)現(xiàn)搜索附近的人具體代碼,供大家參考,具體內(nèi)容如下
實(shí)現(xiàn)思路:
首先,我們應(yīng)該這樣想: 既然我們知道了用戶當(dāng)前位置的經(jīng)緯度,又知道我們將要搜索的范圍,我們可不可以計(jì)算出一個(gè)范圍 ?也就是說,根據(jù)一個(gè)中心點(diǎn)和半徑,計(jì)算出符合條件的經(jīng)緯度的最大值和最小值 。
具體實(shí)現(xiàn):
那么到此,想要獨(dú)立思考完成的小伙伴可以不要繼續(xù)往下看了。
上面我們提到該功能的一個(gè)實(shí)現(xiàn)原理,接下來我們就講解一下具體的實(shí)現(xiàn)步驟。
我們先聲明一個(gè)函數(shù),用作計(jì)算經(jīng)緯度的范圍:
/** * 根據(jù)經(jīng)緯度和半徑計(jì)算出范圍 * @param string $lat 緯度 * @param String $lng 經(jīng)度 * @param float $radius 半徑 * @return Array 范圍數(shù)組 */ private function calcScope($lat, $lng, $radius) { $degree = (24901*1609)/360.0; $dpmLat = 1/$degree; $radiusLat = $dpmLat*$radius; $minLat = $lat - $radiusLat; // 最小緯度 $maxLat = $lat + $radiusLat; // 最大緯度 $mpdLng = $degree*cos($lat * (PI/180)); $dpmLng = 1 / $mpdLng; $radiusLng = $dpmLng*$radius; $minLng = $lng - $radiusLng; // 最小經(jīng)度 $maxLng = $lng + $radiusLng; // 最大經(jīng)度 /** 返回范圍數(shù)組 */ $scope = array( 'minLat' => $minLat, 'maxLat' => $maxLat, 'minLng' => $minLng, 'maxLng' => $maxLng ); return $scope; }
返回的數(shù)組中包含了在 $radius 范圍內(nèi),符合條件的最大最小經(jīng)緯度。
既然我們已經(jīng)獲取到了范圍,那么我們就可以開始從數(shù)據(jù)庫(kù)中查找所有在這個(gè)經(jīng)緯度范圍內(nèi)符合條件的記錄:
/** * 根據(jù)經(jīng)緯度和半徑查詢?cè)诖朔秶鷥?nèi)的所有的電站 * @param String $lat 緯度 * @param String $lng 經(jīng)度 * @param float $radius 半徑 * @return Array 計(jì)算出來的結(jié)果 */ public function searchByLatAndLng($lat, $lng, $radius) { $scope = $this->calcScope($lat, $lng, $radius); // 調(diào)用范圍計(jì)算函數(shù),獲取最大最小經(jīng)緯度 /** 查詢經(jīng)緯度在 $radius 范圍內(nèi)的電站的詳細(xì)地址 */ $sql = 'SELECT `字段` FROM `表名` WHERE `Latitude` < '.$scope['maxLat'].' and `Latitude` > '.$scope['minLat'].' and `Longitude` < '.$scope['maxLng'].' and `Longitude` > '.$scope['minLng']; $stmt = self::$db->query($sql); $res = $stmt->fetchAll(PDO::FETCH_ASSOC); // 獲取查詢結(jié)果并返回 return $res; }
擴(kuò)展:
直到現(xiàn)在,我們已經(jīng)知道了如何計(jì)算出附近的人,但在實(shí)際需求中,我們往往需要計(jì)算出每一個(gè)人與當(dāng)前中心點(diǎn)的實(shí)際距離。
接著,我們?cè)賮砜匆粋€(gè)方法:
/** * 獲取兩個(gè)經(jīng)緯度之間的距離 * @param string $lat1 緯一 * @param String $lng1 經(jīng)一 * @param String $lat2 緯二 * @param String $lng2 經(jīng)二 * @return float 返回兩點(diǎn)之間的距離 */ public function calcDistance($lat1, $lng1, $lat2, $lng2) { /** 轉(zhuǎn)換數(shù)據(jù)類型為 double */ $lat1 = doubleval($lat1); $lng1 = doubleval($lng1); $lat2 = doubleval($lat2); $lng2 = doubleval($lng2); /** 以下算法是 Google 出來的,與大多數(shù)經(jīng)緯度計(jì)算工具結(jié)果一致 */ $theta = $lng1 - $lng2; $dist = sin(deg2rad($lat1)) * sin(deg2rad($lat2)) + cos(deg2rad($lat1)) * cos(deg2rad($lat2)) * cos(deg2rad($theta)); $dist = acos($dist); $dist = rad2deg($dist); $miles = $dist * 60 * 1.1515; return ($miles * 1.609344); }
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
php實(shí)現(xiàn)對(duì)短信驗(yàn)證碼發(fā)送次數(shù)的限制實(shí)例講解
這篇文章主要介紹了php實(shí)現(xiàn)對(duì)短信驗(yàn)證碼發(fā)送次數(shù)的限制實(shí)例講解,案例中列舉了具體代碼實(shí)現(xiàn),有感興趣的同學(xué)可以學(xué)習(xí)下2021-03-03詳解PHP中instanceof關(guān)鍵字及instanceof關(guān)鍵字有什么作用
這篇文章主要介紹了PHP中instanceof關(guān)鍵字詳解及instanceof關(guān)鍵字有什么作用的相關(guān)資料,需要的朋友可以參考下2015-11-11Laravel中任務(wù)調(diào)度console使用方法小結(jié)
這篇文章主要給大家簡(jiǎn)單介紹了Laravel中任務(wù)調(diào)度console使用方法,并附上一個(gè)簡(jiǎn)單的示例,希望對(duì)大家學(xué)習(xí)使用console能夠有所幫助2017-05-05用PHP提取中英文詞語(yǔ)以及數(shù)字的首字母的方法介紹
本篇我在小編為大家介紹,用PHP提取中英文詞語(yǔ)以及數(shù)字的首字母的方法。需要的朋友參考下2013-04-04php 從一個(gè)數(shù)組中隨機(jī)的取出若干個(gè)不同的數(shù)實(shí)例
本文章向碼農(nóng)介紹php從一個(gè)不重復(fù)的數(shù)組中隨機(jī)的取出若干個(gè)不同的元素,難點(diǎn)是防止在取數(shù)的時(shí)候出現(xiàn)已經(jīng)取到過的情況(特別是取到最后),需要盡可能的降低碰撞,需要的朋友可以參考下2016-12-12dedecms 批量提取第一張圖片最為縮略圖的代碼(文章+軟件)
用了dedecms自帶的縮略圖批量獲取,因?yàn)榇a好像是只適用于文字,對(duì)于軟件就不能了,或兩者都不能。從網(wǎng)站看到了這段代碼,對(duì)于dedecms 5.1是沒問題的。2009-10-10Laravel5.5 數(shù)據(jù)庫(kù)遷移:創(chuàng)建表與修改表示例
今天小編就為大家分享一篇Laravel5.5 數(shù)據(jù)庫(kù)遷移:創(chuàng)建表與修改表示例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2019-10-10