声明:本站文章均为作者个人原创,图片均为实际截图。如有需要请收藏网站,禁止转载,谢谢配合!!!

本篇博客已配套视频讲解教程, 点击查看 Bilibili 视频教程



实际开发中,常常需要获取用户附近的商家,思路是

  • 获取用户位置(经纬度信息)
  • 在数据库中查询在距离范围内的商家

注: 本文章内计算距离所使用地球半径统一为 6378.138 km

思路一

直接在数据库中计算经纬度距离并排序

1.在数据库中查询数据

userLatitudeuserLongitude 分别是用户的纬度和经度

latitudelongitude 分别是数据库中商家的纬度和经度字段

    //组合距离sql
    $distanceSql = "(6378.138 * 2 * asin(sqrt(pow(sin((latitude * pi() / 180 - ".$latitude." * pi() / 180) / 2),2) + cos(latitude * pi() / 180) * cos(".$userLatitude." * pi() / 180) * pow(sin((longitude * pi() / 180 - ".$userLongitude." * pi() / 180) / 2),2))) * 1000)";

    //查询
    ShopModel::field("id,name,{$distanceSql} as distance")
        ->where(Db::raw($distanceSql . "<=" . ($distance*1000)))  //获取多少范围之内
        ->order('distance') //按照距离排序
        ->paginate(10)
        ->each(function ($item){
            $item['distance'] = $this->formatDistance($item['distance']);
            return $item;
        });

注: 获取多少范围之内需要用原生的 ->where(Db::raw($distanceSql . "<=" . ($distance*1000)))

不能用 ->where($distanceSql, "<=" . ($distance*1000)), 因为长表达式后面的条件不识别

2.处理数据

第一步得到的距离distance的单位是 米(m), 如果大于1000则显示 千米(km)

public function formatDistance($distance){
    if ($distance > 1000) return round($distance / 1000, 2) . 'km';
    return round($distance, 2) . 'm';
}

3.实际测试

0.5km范围内的商家

图片alt

1km范围内的商家

图片alt

2km范围内的商家

图片alt

思路二

查找并计算

    public function getListByDistance2($longitude, $latitude, $distance){
        //1.计算最大最小经纬度范围
        $range  = 180 / pi() * $distance / 6378.138; //搜索 N km 之内
        $lngR   = $range / cos($latitude * pi() / 180);
        $maxLat = $latitude + $range; //最大纬度
        $minLat = $latitude - $range; //最小纬度
        $maxLng = $longitude + $lngR; //最大经度
        $minLng = $longitude - $lngR; //最小经度
        //2.查找经纬度符合条件的商家
        $list = ShopModel::field("id,name,longitude,latitude")
                ->whereBetween('latitude', [$minLat, $maxLat])
                ->whereBetween('longitude', [$minLng, $maxLng])
                ->where('status', 1)
                ->select();
        //3.计算距离
        foreach ($list as &$item){
            $item['distance'] = $this->getDistanceBy2Point([$longitude, $latitude], [$item['longitude'], $item['latitude']]);
        }

        //4.排序
        $list = CommonUtil::arraySort($list, 'distance');

        return $list;
    }

二维数组排序方法

 public static function arraySort($arr, $field, $sort = SORT_ASC){
        $key = array_column($arr, $field);
        array_multisort($key, $sort, $arr);
        return $arr;
}

根据经纬度计算两点距离

 /**
     * 根据起点坐标和终点坐标测距离
     * @param  [array]   $from     [起点坐标(经纬度),例如:array(118.012951,36.810024)]
     * @param  [array]   $to     [终点坐标(经纬度)]
     * @param  [bool]    $km        是否以公里为单位 false:米 true:公里(千米)
     * @param  [int]     $decimal   精度 保留小数位数
     * @return [string]  距离数值
     */
    function getDistanceBy2Point($from, $to, $km = true, $decimal = 2){
        sort($from);
        sort($to);
        $EARTH_RADIUS = 6378.138; // 地球半径系数

        $distance = $EARTH_RADIUS*2*asin(sqrt(pow(sin( ($from[0]*pi()/180-$to[0]*pi()/180)/2),2)+cos($from[0]*pi()/180)*cos($to[0]*pi()/180)* pow(sin( ($from[1]*pi()/180-$to[1]*pi()/180)/2),2)))*1000;

        if($km && $distance > 1000){
            return round($distance / 1000, 2) . 'km';
        }

        return round($distance, $decimal) . 'm';
    }

实际测试

0.5km范围内的商家

图片alt

1km范围内的商家

图片alt

2km范围内的商家

图片alt

总结

  • 可以看到如果地球半径定为一样的话,两种方法计算出的距离结果是一样的
  • 第一种比较简洁而且支持数据库内分页查询,推荐第一种