목차
요약
백엔드 개발 PHP 튜토리얼 PHP를 사용하여 원하는 주변 사람들을 찾는 방법을 가르쳐주세요.

PHP를 사용하여 원하는 주변 사람들을 찾는 방법을 가르쳐주세요.

Oct 22, 2020 am 11:49 AM
php

최근에 근처 사람을 찾는 비즈니스 시나리오가 있어서 관련 정보를 확인하고 PHP를 사용하여 관련 기능을 구현하는 다양한 방법과 구체적인 구현에 대한 기술 요약을 작성했습니다. 읽어보세요. 댓글과 수정 사항을 올리고 본론으로 들어가겠습니다.

LBS(위치 기반 서비스)

주변 사람 찾기는 LBS(위치 기반 서비스)라는 더 큰 용어로 LBS를 의미합니다. 이동통신 사업자의 무선통신망이나 외부 측위 방법을 통해 이동단말 사용자의 위치정보를 획득하고, GIS 플랫폼의 지원을 받아 사용자에게 해당 서비스를 제공하는 부가서비스이다. 따라서 사용자의 위치를 ​​먼저 획득해야 합니다. 사용자의 위치는 GPS, 사업자 기지국, WIFI 등을 통해 획득할 수 있습니다. 일반적으로 클라이언트는 사용자 위치의 경도 및 위도 좌표를 획득하여 애플리케이션 서버에 업로드합니다. 응용 프로그램 서버는 사용자 좌표를 저장하고 클라이언트는 주변 사람들의 데이터를 얻을 때 데이터베이스로 이동하여 요청자의 지리적 위치와 특정 조건(거리, 성별, 활동 시간 등)을 기반으로 필터링하고 정렬합니다.

경도와 위도를 기준으로 두 점 사이의 거리를 구하는 방법은 무엇입니까?

평면 좌표계에서 두 점의 좌표는 평면 좌표 거리 공식을 사용하여 계산할 수 있다는 것은 우리 모두 알고 있지만 경도와 위도는 3차원 공간의 구를 사용하여 지구상의 공간을 정의하는 구면 좌표계입니다. 지구가 완벽한 구라고 가정할 때, 구면 거리에 관해 계산 공식은 다음과 같습니다.

PHP를 사용하여 원하는 주변 사람들을 찾는 방법을 가르쳐주세요.

구체적인 추론 과정에 관심이 있다면 이 글을 추천합니다: [수학 공식 및 유도] 사이의 거리를 계산합니다. 경도와 위도를 기준으로 지상의 두 점

PHP 함수 코드는 다음과 같습니다.

/**
     * 根据两点间的经纬度计算距离
     * @param $lat1
     * @param $lng1
     * @param $lat2
     * @param $lng2
     * @return float
     */
    public static function getDistance($lat1, $lng1, $lat2, $lng2){
        $earthRadius = 6367000; //approximate radius of earth in meters
        $lat1 = ($lat1 * pi() ) / 180;
        $lng1 = ($lng1 * pi() ) / 180;
        $lat2 = ($lat2 * pi() ) / 180;
        $lng2 = ($lng2 * pi() ) / 180;
        $calcLongitude = $lng2 - $lng1;
        $calcLatitude = $lat2 - $lat1;
        $stepOne = pow(sin($calcLatitude / 2), 2) + cos($lat1) * cos($lat2) * pow(sin($calcLongitude / 2), 2);
        $stepTwo = 2 * asin(min(1, sqrt($stepOne)));
        $calculatedDistance = $earthRadius * $stepTwo;
        return round($calculatedDistance);
    }
로그인 후 복사

MySQL 코드는 다음과 같습니다.

SELECT  
  id, (  
    3959 * acos (  
      cos ( radians(78.3232) )  
      * cos( radians( lat ) )  
      * cos( radians( lng ) - radians(65.3234) )  
      + sin ( radians(78.3232) )  
      * sin( radians( lat ) )  
    )  
  ) AS distance  
FROM markers  
HAVING distance < 30  
ORDER BY distance  
LIMIT 0 , 20;
로그인 후 복사

위의 구면 거리 공식을 계산하여 구하는 것 외에도 다음을 사용할 수 있습니다. Redis 및 MongoDB와 같은 특정 데이터베이스 서비스:

Redis 3.2는 두 위치 사이의 거리를 얻을 수 있을 뿐만 아니라 위치 범위 내에서 지리적 위치 모음을 쉽게 지정할 수 있는 GEO 지리적 위치 기능을 제공합니다. Redis 명령 문서

1. 지리적 위치 추가

GEOADD key longitude latitude member [longitude latitude member ...]
로그인 후 복사

2. 지리적 위치 가져오기

GEOPOS key member [member ...]
로그인 후 복사

3. 두 지리적 위치 사이의 거리를 가져옵니다.

GEODIST key member1 member2 [unit]
로그인 후 복사

4. latitude

GEORADIUS key longitude latitude radius m|km|ft|mi [WITHCOORD] [WITHDIST] [WITHHASH] [COUNT count] [ASC|DESC] [STORE key] [STOREDIST key]
로그인 후 복사

5. 특정 회원의 지리정보 위치 수집을 가져옵니다

GEORADIUSBYMEMBER key member radius m|km|ft|mi [WITHCOORD] [WITHDIST] [WITHHASH] [COUNT count] [ASC|DESC] [STORE key] [STOREDIST key]
로그인 후 복사

MongoDB는 이러한 종류의 쿼리를 위해 특별히 지리공간 인덱스를 구축했습니다. 2d 및 2dsphere 인덱스는 각각 평면과 구에 대한 것입니다. MongoDB 문서

1. 데이터 추가

db.location.insert( {uin : 1 , loc : { lon : 50 , lat : 50 } } )
로그인 후 복사

2. 인덱스 구축

db.location.ensureIndex( { loc : "2d" } )
로그인 후 복사

4. 최대 거리 및 항목 수 제한

db.location.find( { loc :{ $near : [50, 50] } )
로그인 후 복사

5. 포인트와 쿼리 포인트 사이의 거리

db.location.find( { loc : { $near : [50, 50] , $maxDistance : 5 } } ).limit(20)
로그인 후 복사

6. 쿼리 조건 및 반환 번호와 함께 geoNear를 사용하세요. GeoNear는 runCommand 명령을 사용하며 찾기 쿼리에서 페이징과 관련된 매개 변수 제한 및 건너뛰기 기능을 지원하지 않습니다.

db.runCommand( { geoNear : "location" , near : [ 50 , 50 ], num : 10, query : { type : "museum" } } )
로그인 후 복사

PHP 다양한 방법 및 특정 구현

1. MySql 기반

회원 추가 방법:

db.runCommand( { geoNear : "location" , near : [ 50 , 50 ], num : 10, query : { uin : 1 } })
로그인 후 복사

주변 사람 쿼리(쿼리 조건 및 페이징 지원):

public function geoAdd($uin, $lon, $lat)
{
    $pdo = $this->getPdo();
    $sql = &#39;INSERT INTO `markers`(`uin`, `lon`, `lat`) VALUES (?, ?, ?)&#39;;
    $stmt = $pdo->prepare($sql);
    return $stmt->execute(array($uin, $lon, $lat));
}
로그인 후 복사

2 Redis(3.2 이상) 기반

PHP 가능 Redis를 사용하여

redis

확장을 설치하거나 작곡가를 통해 predis 클래스 라이브러리를 설치합니다. 이 문서에서는 이를 구현하기 위해 redis 확장을 사용합니다. 회원 추가 방법:

public function geoNearFind($lon, $lat, $maxDistance = 0, $where = array(), $page = 0)
{
    $pdo = $this->getPdo();
    $sql = "SELECT  
              id, (  
                3959 * acos (  
                  cos ( radians(:lat) )  
                  * cos( radians( lat ) )  
                  * cos( radians( lon ) - radians(:lon) )  
                  + sin ( radians(:lat) )  
                  * sin( radians( lat ) )  
                )  
              ) AS distance  
            FROM markers";

    $input[&#39;:lat&#39;] = $lat;
    $input[&#39;:lon&#39;] = $lon;

    if ($where) {
        $sqlWhere = &#39; WHERE &#39;;
        foreach ($where as $key => $value) {
            $sqlWhere .= "`{$key}` = :{$key} ,";
            $input[":{$key}"] = $value;
        }
        $sql .= rtrim($sqlWhere, &#39;,&#39;);
    }

    if ($maxDistance) {
        $sqlHaving = " HAVING distance < :maxDistance";
        $sql .= $sqlHaving;
        $input[&#39;:maxDistance&#39;] = $maxDistance;
    }

    $sql .= &#39; ORDER BY distance&#39;;

    if ($page) {
        $page > 1 ? $offset = ($page - 1) * $this->pageCount : $offset = 0;
        $sqlLimit = " LIMIT {$offset} , {$this->pageCount}";
        $sql .= $sqlLimit;
    }

    $stmt = $pdo->prepare($sql);
    $stmt->execute($input);
    $list = $stmt->fetchAll(PDO::FETCH_ASSOC);

    return $list;
}
로그인 후 복사

주변 사람 쿼리(쿼리 조건 및 페이징은 지원되지 않음):

public function geoAdd($uin, $lon, $lat)
{
    $redis = $this->getRedis();
    $redis->geoAdd(&#39;markers&#39;, $lon, $lat, $uin);
    return true;
}
로그인 후 복사

3. MongoDB 기반

MongoDB를 사용하는 PHP 확장에는

mongo

(Document) 및 mongodb이 있습니다. ( Documentation) 둘의 작성 방법이 많이 다릅니다. 좋은 확장자를 선택하려면 해당 문서를 확인해야 합니다. mongodb 확장자는 새로운 버전이므로 이 기사에서는 mongodb 확장자를 선택합니다. DB 라이브러리와 위치 컬렉션을 생성한다고 가정하자

인덱스 설정:

public function geoNearFind($uin, $maxDistance = 0, $unit = &#39;km&#39;)
{
    $redis = $this->getRedis();
    $options = [&#39;WITHDIST&#39;]; //显示距离
    $list = $redis->geoRadiusByMember(&#39;markers&#39;, $uin, $maxDistance, $unit, $options);
    return $list;
}
로그인 후 복사

멤버 추가 방법:

db.getCollection(&#39;location&#39;).ensureIndex({"uin":1},{"unique":true}) 
db.getCollection(&#39;location&#39;).ensureIndex({loc:"2d"})
#若查询位置附带查询,可以将常查询条件添加至组合索引
#db.getCollection(&#39;location&#39;).ensureIndex({loc:"2d",uin:1})
로그인 후 복사

주변 사람 쿼리(반환 결과에는 거리가 없고 쿼리 조건 지원, 페이징 지원)

public function geoAdd($uin, $lon, $lat)
{
    $document = array(
        &#39;uin&#39; => $uin,
        &#39;loc&#39; => array(
            &#39;lon&#39; =>  $lon,
            &#39;lat&#39; =>  $lat,
        ),
    );

    $bulk = new MongoDB\Driver\BulkWrite;
    $bulk->update(
        [&#39;uin&#39; => $uin],
        $document,
        [ &#39;upsert&#39; => true]
    );
    //出现noreply 可以改成确认式写入
    $manager = $this->getMongoManager();
    $writeConcern = new MongoDB\Driver\WriteConcern(1, 100);
    //$writeConcern = new MongoDB\Driver\WriteConcern(MongoDB\Driver\WriteConcern::MAJORITY, 100);
    $result = $manager->executeBulkWrite(&#39;db.location&#39;, $bulk, $writeConcern);

    if ($result->getWriteErrors()) {
        return false;
    }
    return true;
}
로그인 후 복사

주변 사람 쿼리 (거리 포함 결과 반환, 쿼리 조건 지원, 결제 반환 수량, 페이징 지원 안 함):

public function geoNearFind($lon, $lat, $maxDistance = 0, $where = array(), $page = 0)
{
    $filter = array(
        &#39;loc&#39; => array(
            &#39;$near&#39; => array($lon, $lat),
        ),
    );
    if ($maxDistance) {
        $filter[&#39;loc&#39;][&#39;$maxDistance&#39;] = $maxDistance;
    }
    if ($where) {
        $filter = array_merge($filter, $where);
    }
    $options = array();
    if ($page) {
        $page > 1 ? $skip = ($page - 1) * $this->pageCount : $skip = 0;
        $options = [
            &#39;limit&#39; => $this->pageCount,
            &#39;skip&#39; => $skip
        ];
    }

    $query = new MongoDB\Driver\Query($filter, $options);
    $manager = $this->getMongoManager();
    $cursor = $manager->executeQuery(&#39;db.location&#39;, $query);
    $list = $cursor->toArray();
    return $list;
}
로그인 후 복사

참고:

1. 좋은 확장자를 선택하세요. mongo와 mongodb 확장자의 작성 방법은 매우 다릅니다

2. 데이터 쓰기 시 응답이 없으면 쓰기 확인 수준을 확인하세요.

3. find를 사용하여 쿼리한 데이터는 거리를 직접 계산해야 하며, geoNear를 사용하여 쿼리한 데이터는 페이징을 지원하지 않습니다.

4. geoNear는 구형 및 거리 승수 매개변수를 사용하여 km로 변환해야 합니다.

위 데모는 여기에서 클릭할 수 있습니다:demo

요약

위에서는 근처 사람들에게 쿼리하는 기능을 구현하는 세 가지 방법을 소개합니다. 예를 들어, 사용자와 여러 도시 사이의 거리를 쿼리하는 데는 Mysql이면 충분합니다. 실시간으로 빠른 응답과 일반적인 검색 범위가 필요한 경우에는 Redis를 사용할 수 있습니다. 데이터의 양이 많고 여러 속성 필터링 조건이 있으므로 mongo를 사용하는 것이 더 효율적입니다. 위의 내용은 특정 비즈니스를 기반으로 구체적인 구현 계획을 검토해야 합니다.

위 내용은 PHP를 사용하여 원하는 주변 사람들을 찾는 방법을 가르쳐주세요.의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

AI Hentai Generator

AI Hentai Generator

AI Hentai를 무료로 생성하십시오.

인기 기사

R.E.P.O. 에너지 결정과 그들이하는 일 (노란색 크리스탈)
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 최고의 그래픽 설정
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 아무도들을 수없는 경우 오디오를 수정하는 방법
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
WWE 2K25 : Myrise에서 모든 것을 잠금 해제하는 방법
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

Ubuntu 및 Debian용 PHP 8.4 설치 및 업그레이드 가이드 Ubuntu 및 Debian용 PHP 8.4 설치 및 업그레이드 가이드 Dec 24, 2024 pm 04:42 PM

PHP 8.4는 상당한 양의 기능 중단 및 제거를 통해 몇 가지 새로운 기능, 보안 개선 및 성능 개선을 제공합니다. 이 가이드에서는 Ubuntu, Debian 또는 해당 파생 제품에서 PHP 8.4를 설치하거나 PHP 8.4로 업그레이드하는 방법을 설명합니다.

CakePHP 날짜 및 시간 CakePHP 날짜 및 시간 Sep 10, 2024 pm 05:27 PM

cakephp4에서 날짜와 시간을 다루기 위해 사용 가능한 FrozenTime 클래스를 활용하겠습니다.

CakePHP 파일 업로드 CakePHP 파일 업로드 Sep 10, 2024 pm 05:27 PM

파일 업로드 작업을 위해 양식 도우미를 사용할 것입니다. 다음은 파일 업로드의 예입니다.

CakePHP 토론 CakePHP 토론 Sep 10, 2024 pm 05:28 PM

CakePHP는 PHP용 오픈 소스 프레임워크입니다. 이는 애플리케이션을 훨씬 쉽게 개발, 배포 및 유지 관리할 수 있도록 하기 위한 것입니다. CakePHP는 강력하고 이해하기 쉬운 MVC와 유사한 아키텍처를 기반으로 합니다. 모델, 뷰 및 컨트롤러 gu

CakePHP 유효성 검사기 만들기 CakePHP 유효성 검사기 만들기 Sep 10, 2024 pm 05:26 PM

컨트롤러에 다음 두 줄을 추가하면 유효성 검사기를 만들 수 있습니다.

CakePHP 로깅 CakePHP 로깅 Sep 10, 2024 pm 05:26 PM

CakePHP에 로그인하는 것은 매우 쉬운 작업입니다. 한 가지 기능만 사용하면 됩니다. cronjob과 같은 백그라운드 프로세스에 대해 오류, 예외, 사용자 활동, 사용자가 취한 조치를 기록할 수 있습니다. CakePHP에 데이터를 기록하는 것은 쉽습니다. log() 함수는 다음과 같습니다.

PHP 개발을 위해 Visual Studio Code(VS Code)를 설정하는 방법 PHP 개발을 위해 Visual Studio Code(VS Code)를 설정하는 방법 Dec 20, 2024 am 11:31 AM

VS Code라고도 알려진 Visual Studio Code는 모든 주요 운영 체제에서 사용할 수 있는 무료 소스 코드 편집기 또는 통합 개발 환경(IDE)입니다. 다양한 프로그래밍 언어에 대한 대규모 확장 모음을 통해 VS Code는

CakePHP 빠른 가이드 CakePHP 빠른 가이드 Sep 10, 2024 pm 05:27 PM

CakePHP는 오픈 소스 MVC 프레임워크입니다. 이를 통해 애플리케이션 개발, 배포 및 유지 관리가 훨씬 쉬워집니다. CakePHP에는 가장 일반적인 작업의 과부하를 줄이기 위한 여러 라이브러리가 있습니다.

See all articles