之前我们计算地理空间点位之间的距离等操作,是使用mysql自带的函数进行处理,这样的话需要每个操作都要单独编写一个sql语句,非常的麻烦并且时间查询较慢,效率比较低,今天我们使用 Spring Boot + Geodesy 实现地理空间高精度坐标处理及多样化的距离计算功能,更加方便快捷。
这是之前使用mysql进行简单空间操作的文章:mysql中point的使用
Geodesy 是一个专注于地理空间计算的强大框架。它具有以下突出特性:
1.高精度坐标处理:能够精确处理和转换不同的地理坐标系统,确保坐标数据的准确性和一致性。
2.多样化的距离计算方法:支持多种距离计算模式,如直线距离、测地线距离等,满足各种复杂的业务需求。
3.强大的空间分析能力:可以进行空间关系的判断,如点与区域的包含关系、区域之间的相交关系等。
4.高效性能:在处理大规模地理数据时,能保持高效的计算速度和低内存消耗。
5.跨平台兼容性:可以在不同的操作系统和开发环境中稳定运行。
这些特性使得 Geodesy 在地理信息系统、导航应用、物流规划等领域具有广泛的应用价值。在许多应用场景中,需要计算两个地点之间的距离。本文将介绍如何使用 Spring Boot 结合 Geodesy 库来实现距离计算功能。
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0modelVersion>
<parent>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-parentartifactId>
<version>3.0.0version>
<relativePath/>
parent>
<groupId>com.examplegroupId>
<artifactId>distance-calculation-systemartifactId>
<version>0.0.1-SNAPSHOTversion>
<name>Distance Calculation Systemname>
<properties>
<java.version>17java.version>
properties>
<dependencies>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-webartifactId>
dependency>
<dependency>
<groupId>org.gavaghangroupId>
<artifactId>geodesyartifactId>
<version>1.1.3version>
dependency>
dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-maven-pluginartifactId>
plugin>
plugins>
build>
project>
package com.icoderoad.service;
import org.gavaghan.geodesy.Ellipsoid;
import org.gavaghan.geodesy.GeodeticCalculator;
import org.gavaghan.geodesy.GeodeticCurve;
import org.gavaghan.geodesy.GlobalCoordinates;
import java.util.ArrayList;
import java.util.List;
public class GeospatialCalculationService {
//计算直线距离
public double calculateStraightLineDistance(double latitude1, double longitude1, double latitude2, double longitude2) {
GlobalCoordinates point1 = new GlobalCoordinates(latitude1, longitude1);
GlobalCoordinates point2 = new GlobalCoordinates(latitude2, longitude2);
double distance = Math.sqrt(Math.pow((latitude2 - latitude1), 2) + Math.pow((longitude2 - longitude1), 2));
return distance;
}
//计算测地距离(三维空间中最短距离)
public double calculateGeodesicDistance(double latitude1, double longitude1, double latitude2, double longitude2) {
GlobalCoordinates point1 = new GlobalCoordinates(latitude1, longitude1);
GlobalCoordinates point2 = new GlobalCoordinates(latitude2, longitude2);
GeodeticCalculator calculator = new GeodeticCalculator();
GeodeticCurve curve = calculator.calculateGeodeticCurve(Ellipsoid.WGS84, point1, point2);
return curve.getEllipsoidalDistance();
}
//是否为区域内的点
public boolean isPointInRegion(double pointLatitude, double pointLongitude, List<GlobalCoordinates> regionPoints) {
// 假设区域是凸多边形
int n = regionPoints.size();
boolean inside = false;
for (int i = 0, j = n - 1; i < n; j = i++) {
if (((regionPoints.get(i).getLatitude() > pointLatitude)!= (regionPoints.get(j).getLatitude() > pointLatitude))
&& (pointLongitude < (regionPoints.get(j).getLongitude() - regionPoints.get(i).getLongitude())
* (pointLatitude - regionPoints.get(i).getLatitude())
/ (regionPoints.get(j).getLatitude() - regionPoints.get(i).getLatitude())
+ regionPoints.get(i).getLongitude())) {
inside =!inside;
}
}
return inside;
}
//区域是否相交
public boolean isRegionIntersect(List<GlobalCoordinates> region1Points, List<GlobalCoordinates> region2Points) {
// 简单判断,假设两个区域都是凸多边形且边界不相交
for (GlobalCoordinates point1 : region1Points) {
if (isPointInRegion(point1.getLatitude(), point1.getLongitude(), region2Points)) {
return true;
}
}
for (GlobalCoordinates point2 : region2Points) {
if (isPointInRegion(point2.getLatitude(), point2.getLongitude(), region1Points)) {
return true;
}
}
return false;
}
//计算面积
public double calculateArea(List<GlobalCoordinates> polygonPoints) {
double area = 0.0;
int n = polygonPoints.size();
for (int i = 0; i < n; i++) {
int j = (i + 1) % n;
area += polygonPoints.get(i).getLongitude() * polygonPoints.get(j).getLatitude()
- polygonPoints.get(j).getLongitude() * polygonPoints.get(i).getLatitude();
}
area /= 2.0;
return Math.abs(area);
}
}
package com.icoderoad.controller;
import com.icoderoad.service.GeospatialCalculationService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping("/geospatial")
public class GeospatialController {
@Autowired
private GeospatialCalculationService geospatialCalculationService;
//计算直线距离
@GetMapping("/calculateStraightLineDistance")
public double calculateStraightLineDistance(@RequestParam double latitude1, @RequestParam double longitude1,
@RequestParam double latitude2, @RequestParam double longitude2) {
return geospatialCalculationService.calculateStraightLineDistance(latitude1, longitude1, latitude2, longitude2);
}
//计算测地距离(三维空间中最短距离)
@GetMapping("/calculateGeodesicDistance")
public double calculateGeodesicDistance(@RequestParam double latitude1, @RequestParam double longitude1,
@RequestParam double latitude2, @RequestParam double longitude2) {
return geospatialCalculationService.calculateGeodesicDistance(latitude1, longitude1, latitude2, longitude2);
}
//是否为区域内的点
@GetMapping("/isPointInRegion")
public boolean isPointInRegion(@RequestParam double pointLatitude, @RequestParam double pointLongitude,
@RequestParam List<Double> regionLatitudes, @RequestParam List<Double> regionLongitudes) {
List<GlobalCoordinates> regionPoints = new ArrayList<>();
for (int i = 0; i < regionLatitudes.size(); i++) {
regionPoints.add(new GlobalCoordinates(regionLatitudes.get(i), regionLongitudes.get(i)));
}
return geospatialCalculationService.isPointInRegion(pointLatitude, pointLongitude, regionPoints);
}
//区域是否相交
@GetMapping("/isRegionIntersect")
public boolean isRegionIntersect(@RequestParam List<Double> region1Latitudes, @RequestParam List<Double> region1Longitudes,
@RequestParam List<Double> region2Latitudes, @RequestParam List<Double> region2Longitudes) {
List<GlobalCoordinates> region1Points = new ArrayList<>();
List<GlobalCoordinates> region2Points = new ArrayList<>();
for (int i = 0; i < region1Latitudes.size(); i++) {
region1Points.add(new GlobalCoordinates(region1Latitudes.get(i), region1Longitudes.get(i)));
}
for (int i = 0; i < region2Latitudes.size(); i++) {
region2Points.add(new GlobalCoordinates(region2Latitudes.get(i), region2Longitudes.get(i)));
}
return geospatialCalculationService.isRegionIntersect(region1Points, region2Points);
}
//计算面积
@GetMapping("/calculateArea")
public double calculateArea(@RequestParam List<Double> polygonLatitudes, @RequestParam List<Double> polygonLongitudes) {
List<GlobalCoordinates> polygonPoints = new ArrayList<>();
for (int i = 0; i < polygonLatitudes.size(); i++) {
polygonPoints.add(new GlobalCoordinates(polygonLatitudes.get(i), polygonLongitudes.get(i)));
}
return geospatialCalculationService.calculateArea(polygonPoints);
}
}