经纬度距离排序
经纬度距离排序
工具类排序
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)列表。 - 在指定了
WITHCOORD
、WITHDIST
、WITHHASH
等选项的情况下, 命令返回一个二层嵌套数组, 内层的每个子数组就表示一个元素。
在返回嵌套数组时, 子数组的第一个元素总是位置元素的名字。 至于额外的信息, 则会作为子数组的后续元素, 按照以下顺序被返回:
- 以浮点数格式返回的中心与位置元素之间的距离, 单位与用户指定范围时的单位一致。
- geohash 整数。
- 由两个元素组成的坐标,分别为经度和纬度。
GEORADIUSBYMEMBER
它的参数和georadiusbyrnernber基本一致,差别是将经纬度坐标值改成目标元素
上一篇: 深入理解Redis(番外)——持久化