欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

经纬度距离排序

程序员文章站 2022-07-05 18:05:18
...

经纬度距离排序

工具类排序

DistanceUtil

import java.math.BigDecimal;

public class DistanceUtil {
    private static double EARTH_RADIUS = 6378.137;

    private static double rad(double d) {
        return d * Math.PI / 180.0;
    }

    /**
     * 通过经纬度获取距离(单位:千米)
     * @param lat1
     * @param lng1
     * @param lat2
     * @param lng2
     * @return
     */
    public static BigDecimal getDistance(double lat1, double lng1, double lat2,
                                         double lng2) {
        double radLat1 = rad(lat1);
        double radLat2 = rad(lat2);
        double a = radLat1 - radLat2;
        double b = rad(lng1) - rad(lng2);
        double s = 2 * Math.asin(Math.sqrt(Math.pow(Math.sin(a / 2), 2)
                + Math.cos(radLat1) * Math.cos(radLat2)
                * Math.pow(Math.sin(b / 2), 2)));
        s = s * EARTH_RADIUS;
        s = Math.round(s * 10000d) / 10000d;
        BigDecimal bd = new BigDecimal(s);
        //保留小数点后两位
        bd = bd.setScale(2, BigDecimal.ROUND_HALF_UP);
        return bd;
    }
}

利用treeMap排序,默认小到大

List<TbPeripheralFacilities> Facilities = tbPeripheralFacilitiesMapper.selectTbPeripheralFacilitiesH5List(tbPeripheralFacilities);
       Map<BigDecimal,TbPeripheralFacilities> addressMap=new TreeMap<>();
       for (TbPeripheralFacilities facility : Facilities) {
           BigDecimal distance = DistanceUtil.getDistance(Double.valueOf(tbPeripheralFacilities.getLat()),
                   Double.valueOf(tbPeripheralFacilities.getLng()),
                   Double.valueOf(facility.getLat()),
                   Double.valueOf(facility.getLng()));
           addressMap.put(distance,facility);
       }
       List<TbPeripheralFacilities> collect = addressMap.values().stream().collect(Collectors.toList());
       return collect;

redis GeoHash

pom.xml

<dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-redis</artifactId>
        </dependency>

serviceImpl 片段

//把当前用户的身份证作为geoHash的member,存入用户当前经纬度
redisCache.redisTemplate.opsForGeo().add("position", new Point(Double.valueOf(recruitingPosition.getLng()), Double.valueOf(recruitingPosition.getLat())),idCardNumber);
...
RecruitingPosition position = param.getPosition();
Integer pageNum = param.getPageNum();
Integer pageSize = param.getPageSize();
List<RecruitingPosition> scorePositions = recruitingPositionService.selectRecruitingPositionList(position);    
//遍历赋值距离
scorePositions.stream().forEach((e)->{
    Distance distance = redisCache.redisTemplate.opsForGeo().distance("position", e.getId(), param.getIdCardNumber(),RedisGeoCommands.DistanceUnit.KILOMETERS);
    BigDecimal hDistance = new BigDecimal(distance.getValue()).setScale(1, BigDecimal.ROUND_HALF_UP);
    double dDistance = hDistance.compareTo(new BigDecimal(0.1))==-1?0.1:hDistance.doubleValue();
    e.setDistance(String.valueOf(dDistance));
});
//排序
Collections.sort(scorePositions, Comparator.comparing(RecruitingPosition::getDistance,(s1,s2)->{
    return Double.valueOf(s1).compareTo(Double.valueOf(s2));
}).thenComparing(RecruitingPosition::getScore,(t1,t2)->{
    return -t1.compareTo(t2);
}));
//分页
List<RecruitingPosition> resultList = scorePositions.stream().skip(pageSize*(pageNum-1)).limit(pageSize).collect(Collectors.toList());
RecruitingPositionList result =  new RecruitingPositionList();
result.setPositions(resultList);
result.setSize(scorePositions.size());
return result;

GEOADD

命令:GEOADD key longitude latitude member [longitude latitude member …]

命令描述:将指定的地理空间位置(纬度、经度、名称)添加到指定的key中。

返回值:添加到sorted set元素的数目,但不包括已更新score的元素。

GEODIST

命令:GEODIST key member1 member2 [unit]

命令描述:

返回两个给定位置之间的距离。如果两个位置之间的其中一个不存在, 那么命令返回空值。指定单位的参数 unit 必须是以下单位的其中一个:

  • m 表示单位为米。
  • km 表示单位为千米。
  • mi 表示单位为英里。
  • ft 表示单位为英尺。

GEOPOS

命令:GEOPOS key member [member …]

命令描述:从key里返回所有给定位置元素的位置(经度和纬度)。

返回值:GEOPOS 命令返回一个数组, 数组中的每个项都由两个元素组成: 第一个元素为给定位置元素的经度, 而第二个元素则为给定位置元素的纬度。当给定的位置元素不存在时, 对应的数组项为空值。

GEOHASH

命令:GEOHASH key member [member …]

命令描述:返回一个或多个位置元素的 Geohash 表示。通常使用表示位置的元素使用不同的技术,使用Geohash位置52点整数编码。由于编码和解码过程中所使用的初始最小和最大坐标不同,编码的编码也不同于标准。此命令返回一个标准的Geohash

返回值:一个数组, 数组的每个项都是一个 geohash 。 命令返回的 geohash 的位置与用户给定的位置元素的位置一一对应。

GEORADIUS

命令:GEORADIUS key longitude latitude radius m|km|ft|mi [WITHCOORD] [WITHDIST] [WITHHASH] [COUNT count]

命令描述:

以给定的经纬度为中心, 返回键包含的位置元素当中, 与中心的距离不超过给定最大距离的所有位置元素。

范围可以使用以下其中一个单位:

  • m 表示单位为米。
  • km 表示单位为千米。
  • mi 表示单位为英里。
  • ft 表示单位为英尺。

在给定以下可选项时, 命令会返回额外的信息:

  • WITHDIST: 在返回位置元素的同时, 将位置元素与中心之间的距离也一并返回。 距离的单位和用户给定的范围单位保持一致。
  • WITHCOORD: 将位置元素的经度和维度也一并返回。
  • WITHHASH: 以 52 位有符号整数的形式, 返回位置元素经过原始 geohash 编码的有序集合分值。 这个选项主要用于底层应用或者调试, 实际中的作用并不大。

命令默认返回未排序的位置元素。 通过以下两个参数, 用户可以指定被返回位置元素的排序方式:

  • ASC: 根据中心的位置, 按照从近到远的方式返回位置元素。
  • DESC: 根据中心的位置, 按照从远到近的方式返回位置元素。

在默认情况下, GEORADIUS 命令会返回所有匹配的位置元素。 虽然用户可以使用 COUNT <count> 选项去获取前 N 个匹配元素, 但是因为命令在内部可能会需要对所有被匹配的元素进行处理, 所以在对一个非常大的区域进行搜索时, 即使只使用 COUNT 选项去获取少量元素, 命令的执行速度也可能会非常慢。 但是从另一方面来说, 使用 COUNT 选项去减少需要返回的元素数量, 对于减少带宽来说仍然是非常有用的。

返回值:

  • 在没有给定任何 WITH 选项的情况下, 命令只会返回一个像 [“New York”,”Milan”,”Paris”] 这样的线性(linear)列表。
  • 在指定了 WITHCOORDWITHDISTWITHHASH 等选项的情况下, 命令返回一个二层嵌套数组, 内层的每个子数组就表示一个元素。

在返回嵌套数组时, 子数组的第一个元素总是位置元素的名字。 至于额外的信息, 则会作为子数组的后续元素, 按照以下顺序被返回:

  1. 以浮点数格式返回的中心与位置元素之间的距离, 单位与用户指定范围时的单位一致。
  2. geohash 整数。
  3. 由两个元素组成的坐标,分别为经度和纬度。

GEORADIUSBYMEMBER

它的参数和georadiusbyrnernber基本一致,差别是将经纬度坐标值改成目标元素

相关标签: # 方法 java