php随机经纬度,PHP轻松实现"附近的人"功能,根据IP确定经纬度,根据经纬度计算距离...

PHP安装GeoIP扩展和数据库根据IP获取访客所在国家/城市/经纬度等信息

然后就可以用geoip_record_by_name($_SERVER['REMOTE_ADDR'])根据用户IP确定经纬度了.

注意:geoip_record_by_name()返回的西经和南纬是负数.

5000米转成经纬度:

纬度 Latitude:  1 deg = 110852 m

经度 Longitude: 1 deg = 111320*cos(lat) m

同一经线上,相差一纬度约为 110852 米

同一纬线上,相差一经度约为 111320*cos(lat) 米 (lat为该纬线的纬度)

//以当前用户经纬度为中心,查询5000米内的其他用户

$y = 5000 / 110852; //纬度的范围

$x = 5000 / (111320*cos($lat)); //经度的范围

$sql = '

select * from user where

lat >= ($lat-$y) and lat <= ($lat+$y) and

lon >= ($lon-$x) and lon <= ($lon+$x);

';

数据库用户表中设两个字段,分别存储用户的经度lat和纬度lon.

($lat-$y)  <= lat <= ($lat+$y)

($lon-$x)  <= lon <= ($lon+$x)

这个范围是一个粗略的范围,下面计算距离后把超过5公里的用户去掉即可.

根据上面查询出来的用户的经纬度,

用半正矢公式(Haversine)根据经纬度计算两点间距离:

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由近到远列出用户了,比如有名为win,osx,lin这3个用户:

$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}}

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值