ThinkPHP+MySQL查询数据的时候计算两个经纬度之间的距离

需求,数据表中有lng(经度)lat(维度)两个字段,查询数据的时候要计算记录经纬度距离目标经纬度之间的距离。

方法中还有根据生日计算年龄(YEAR(CURDATE()) - YEAR(birthday)) AS age

    public function get_lists(){
        $request = $this->request->get();
//获取前端传来的目标经纬度
        if(empty($request['lng']) || empty($request['lat'])){
            $this->error('位置信息错误');
        }
        $where = [];
        $lists = PurposeModel::alias('p')
            ->join('user u','p.user_id = u.id')
            ->join('resume r','p.user_id = r.user_id')
            ->join('category c','p.category_id = c.id')
            ->where($where)
            //根据生日计算年龄(YEAR(CURDATE()) - YEAR(birthday)) AS age 根据经纬度计算距离
            ->field('p.user_id,r.name,u.avatar,(YEAR(CURDATE()) - YEAR(birthday)) AS age,r.sex,c.name as category_name
                        ,p.gongzhong,p.work_begin,p.work_end,p.rest_begint,p.rest_end,p.city,p.address,'.getDistanceBuilder($request['lat'], $request['lng']))
            ->paginate();
        $this->success('获取成功',$lists);
    }

在common.php中添加下面的方法

if (!function_exists('getDistanceBuilder')) {
    /**
     * 数据库查询的时候计算两点距离,命名为 distance 字段
     * @param string $lat 维度
     * @param string $lng 经度
     * 返回的距离字段 distance 单位是米
     * p.lat是我的表别名和维度字段名
     * p.lng是我的表别名和经度字段名
     * 字段名要根据自己的表的别名进行修改
     * @return string
     */
    function getDistanceBuilder($lat, $lng) {
        return "ROUND(6378.138 * 2 * ASIN(SQRT(POW(SIN((". matchLatLng($lat) . " * PI() / 180 - p.lat * PI() / 180) / 2), 2) + COS(". matchLatLng($lat). " * PI() / 180) * COS(p.lat * PI() / 180) * POW(SIN((". matchLatLng($lng). " * PI() / 180 - p.lng * PI() / 180) / 2), 2))) * 1000) AS distance";
    }
}

if (!function_exists('matchLatLng')) {
    function matchLatLng($latlng) {
        $match = "/^\d{1,3}\.\d{1,30}$/";
        return preg_match($match, $latlng) ? $latlng : 0;
    }
}

代码就可以实现在查询结果中增加distance列用于记录两个经纬度坐标之间的距离,可该字段进行排序。

  • 6
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值