言成言成啊 | Kit Chen's Blog

Geometry切块-根据是否有交点或者被包含

发布于2022-05-15 02:44:39,更新于2023-01-17 22:56:27,标签:java gis  文章会持续修订,转载请注明来源地址:https://meethigher.top/blog

一、废话

介绍正文之前,先记录下周六玩的东西。苏见信的《顽魂》是真的好听,降噪耳机,单曲循环24小时+码字,俩字浪漫

“悬崖上花开,半空仍纯白,逆境中有什么不能重来,天地的熔炉,烧炼我凡胎,烧不尽顽固血脉,别想我敬拜”

“尽管烧成埃,以血肉灭黑白,凭一身顽魂不会坏”

直接抓包下了MV,想自己打磨下——提高分辨率补帧,这里发现了几个好用的算法/工具。

  1. ailab: B站开源算法,超吃显存,我的1650显存4G带不动,不过出来的效果吊打下面两个,只能说B站调教还是牛逼
  2. Squirrel-RIFE: 效果更好的补帧软件,显存占用更小,是DAIN速度的10-25倍,包含抽帧处理,去除动漫卡顿感
  3. Waifu2x-Extension-GUI: 提高分辨率+补帧的GUI工具,好用,显存占用少,但是默认引擎出来效果垃圾,当然了可以选用B站的引擎

丢到B站上去了,【完美世界】aiLab渲染的纯享版《顽魂》_哔哩哔哩_bilibili,瑕疵还是有的吧,也就60帧还可以,不过没有B站大会员也体会不到,哈哈。

二、正文

老规矩,放出抄袭来源!

  1. Vector grids — GeoTools 27-SNAPSHOT User Guide
  2. clydedacruz/openstreetmap-wkt-playground: Plot and visualize WKT shapes on OpenStreetMap
  3. asapelkin/wkt_3d_viewer: WKT (Well-known text) viewer on the 3D Earth
  4. wkt在线绘制展示_EPSG4326
  5. meethigher/wkt-show-on-openlayers: 基于openlayers的wkt绘制展示功能
  6. 本文源码

工作中的一个需求,功能就是落点分析。要精确的分析,一开始使用的缓存拓宽后合并,然后根据合并的结果是否包含指定区域,来确定是否完全落点。但是要精确到某个地方没落点就不行了。于是公司大佬就研究下了geotools实现了一个分块功能,根据分块来落点,然后最后取出没落点的块,实现漏点分析。

趁着周末学习一下大佬的写法,发现大佬的写法也是有Bug的,于是便想着优化一下。

先不扯大佬怎么写的,自己比着文档一步一步来。

2.1 依赖

拉取依赖如果出问题,可以参照我之前的文章

maven依赖pom

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
<?xml version="1.0" encoding="UTF-8"?>
<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 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>

<groupId>top.meethigher.geometrygrid</groupId>
<artifactId>geometry-grid</artifactId>
<version>1.0.0</version>

<properties>
<java.version>8</java.version>
<maven.compiler.source>1.8</maven.compiler.source>
<maven.compiler.target>1.8</maven.compiler.target>
<project.build.sourceEncoding>utf-8</project.build.sourceEncoding>
<compiler.encoding>utf-8</compiler.encoding>
<geotools.version>22-RC</geotools.version>
<slf4j.version>1.7.36</slf4j.version>
<logback.version>1.2.3</logback.version>
</properties>

<repositories>
<repository>
<id>osgeo</id>
<name>OSGeo Release Repository</name>
<url>https://repo.osgeo.org/repository/release/</url>
<snapshots>
<enabled>false</enabled>
</snapshots>
<releases>
<enabled>true</enabled>
</releases>
</repository>
</repositories>

<dependencies>
<!--[slf4j logback配置](http://www.51gjie.com/javaweb/1123.html)-->
<!--slf4j日志-->
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-api</artifactId>
<version>${slf4j.version}</version>
</dependency>
<!--logback-->
<dependency>
<groupId>ch.qos.logback</groupId>
<artifactId>logback-classic</artifactId>
<version>${logback.version}</version>
</dependency>
<dependency>
<groupId>ch.qos.logback</groupId>
<artifactId>logback-core</artifactId>
<version>${logback.version}</version>
</dependency>
<dependency>
<groupId>org.geotools</groupId>
<artifactId>gt-grid</artifactId>
<version>${geotools.version}</version>
</dependency>
<dependency>
<groupId>org.geotools</groupId>
<artifactId>gt-opengis</artifactId>
<version>${geotools.version}</version>
</dependency>


<dependency>
<groupId>org.geotools</groupId>
<artifactId>gt-epsg-hsql</artifactId>
<version>${geotools.version}</version>
<exclusions>
<exclusion>
<groupId>javax</groupId>
<artifactId>javaee-api</artifactId>
</exclusion>
</exclusions>
</dependency>
<dependency>
<groupId>org.geotools</groupId>
<artifactId>gt-main</artifactId>
<version>${geotools.version}</version>
<exclusions>
<exclusion>
<groupId>javax</groupId>
<artifactId>javaee-api</artifactId>
</exclusion>
</exclusions>
</dependency>

<dependency>
<groupId>org.geotools</groupId>
<artifactId>gt-geojson</artifactId>
<version>${geotools.version}</version>
<exclusions>
<exclusion>
<groupId>javax</groupId>
<artifactId>javaee-api</artifactId>
</exclusion>
</exclusions>
</dependency>

<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
<scope>test</scope>
</dependency>
</dependencies>
</project>

2.2 实现

切面的思路

  1. 以东西为x轴、南北为y轴,获取边界,构建超大矩形
  2. 根据要切的边长,将矩形切成小块。

切线的思路

  1. 获取线的顶点
  2. 每两对顶点构成一个向量,然后从起点根据三角函数,算出所有均分点,切成小块后,旋转角度即可。

下面展示Geotools如何获取边界

1
Geometry envelope = geometry.getEnvelope();

下面展示官方文档给的例子,其实结果已经接近我所想要的90%了。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
/**
* 单元测试
*
* @author chenchuancheng github.com/meethigher
* @since 2022/5/15 22:08
*/
public class GeometryGridTest {

@Test
public void testGrid() throws Exception {

double sideLen = 0.001;

//String wkt = "LINESTRING(106.67105340852869 29.530636809322942,106.66803216986591 29.52328796943324)";
//String wkt = "LINESTRING(106.66641855239867 29.52517003980182,106.67002344131468 29.532727843679083,106.67507028736871 29.530487446637224,106.67675256676739 29.533175917132013)";
String wkt = "POLYGON((106.65735483169556 29.530158851923517,106.65258264620205 29.52902369502023,106.64671182632448 29.530965403423963,106.64609384562937 29.52785865163041,106.64588785250089 29.524781676802462,106.65330362477108 29.522899600833085,106.6530289655202 29.524542686760086,106.65141534831493 29.527111822862878,106.6533379565226 29.527380681416773,106.65756082482405 29.526514355331983,106.65776682057185 29.52403482645859,106.66192102537026 29.522122860471057,106.67208337731424 29.521585112423097,106.67551660747264 29.522929476044652,106.67881250276693 29.52531940968403,106.67943048477169 29.528814585054363,106.68025446252427 29.532459000021845,106.67874383926389 29.53428115740634,106.6783661873778 29.532548613742065,106.67850351438389 29.530218596454873,106.6765122413635 29.531114763609096,106.67575693235264 29.5339824452562,106.67469262972004 29.536192894729282,106.67187738418576 29.53732797116615,106.67012643866471 29.53583444659172,106.67246103443901 29.5338330889642,106.67290735244748 29.531831692071208,106.66882181115214 29.53189143550047,106.66703653387954 29.53278758783246,106.66264200158184 29.535655222388314,106.65986108779906 29.536909786679814,106.65821314015191 29.5349084499357,106.65838480152888 29.532309639998616,106.65735483169556 29.530158851923517))";

WKTReader reader = new WKTReader();
Geometry geometry = reader.read(wkt);

//以东西为x轴,南北为y轴,获取包含此几何图形中最小和最大x和y值
//如果是一条斜线,就重组x、y坐标构成一个矩形。
Envelope envelopeInternal = geometry.getEnvelopeInternal();
double maxX = envelopeInternal.getMaxX();
double minX = envelopeInternal.getMinX();
double maxY = envelopeInternal.getMaxY();
double minY = envelopeInternal.getMinY();

//构建边界,坐标系随便传啦。
//这个坐标系其实也无所谓,作用就是创建的时候,塞进去,读取的时候在查出来。你乱传都不影响结果
//没啥作用,要非要说作用,那就是起个标识的作用啦,方便你断点的时候知道这是个啥
ReferencedEnvelope gridBounds =
new ReferencedEnvelope(minX, maxX, minY, maxY, DefaultGeographicCRS.WGS84);
SimpleFeatureSource grid = Grids.createSquareGrid(gridBounds, sideLen);
List<String> gridList = new LinkedList<>();
SimpleFeatureIterator iterator = grid.getFeatures().features();
while (iterator.hasNext()) {
SimpleFeature feature = iterator.next();
Object defaultGeometry = feature.getDefaultGeometry();
gridList.add(defaultGeometry.toString());
}
gridList.add(wkt);
System.out.println(gridList);
}
}

这三个wkt,最后的结果依次是下面的三张图。

由图可知,就取第一张图来说,根据我的需求,我应该要的是下面这种效果。白色表示丢弃的部分。过滤掉白色这部分,线只需要通过相交判断,如果是面,还需要包含才可。

下面进行修改,修改的过程还是要多断点查看,从createSquareGrid断点进入,查看里面的具体逻辑。此处不多赘述!

关键的地方如下,通过此处我们可以进行过滤。

接下来,自己实现一个Builder,添加自己的逻辑,其他的直接源码拷一份!

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
import org.geotools.feature.simple.SimpleFeatureTypeBuilder;
import org.geotools.grid.GridElement;
import org.geotools.grid.GridFeatureBuilder;
import org.locationtech.jts.geom.Geometry;
import org.locationtech.jts.geom.Polygon;
import org.opengis.feature.simple.SimpleFeatureType;
import org.opengis.referencing.crs.CoordinateReferenceSystem;


import java.util.Map;

/**
* 自定义实现,附加切块的条件过滤
*
* @author chenchuancheng github.com/meethigher
* @since 2022/5/15 21:40
*/
public class GridCreatorFeatureBuilder extends GridFeatureBuilder {


private final Geometry originGeometry;

/**
* Creates a {@code GridFeatureBuilder} to work with the given feature type.
*
* @param type the feature type
* @param originGeometry
*/
public GridCreatorFeatureBuilder(SimpleFeatureType type, Geometry originGeometry) {
super(type);
this.originGeometry = originGeometry;
}

/** Default feature TYPE name: "grid" */
public static final String DEFAULT_TYPE_NAME = "grid";

/** Name used for the integer id attribute: "id" */
public static final String ID_ATTRIBUTE_NAME = "id";

private int id;

/**
* Creates the feature TYPE
*
* @param typeName name for the feature TYPE; if {@code null} or empty, {@linkplain
* #DEFAULT_TYPE_NAME} will be used
* @param crs coordinate reference system (may be {@code null})
* @return the feature TYPE
*/
protected static SimpleFeatureType createType(String typeName, CoordinateReferenceSystem crs) {
final String finalName;
if (typeName != null && typeName.trim().length() > 0) {
finalName = typeName;
} else {
finalName = DEFAULT_TYPE_NAME;
}

SimpleFeatureTypeBuilder tb = new SimpleFeatureTypeBuilder();
tb.setName(finalName);
tb.add(DEFAULT_GEOMETRY_ATTRIBUTE_NAME, Polygon.class, crs);
tb.add("id", Integer.class);
return tb.buildFeatureType();
}

public GridCreatorFeatureBuilder(Geometry originGeometry,CoordinateReferenceSystem crs) {
super(createType(DEFAULT_TYPE_NAME, crs));
this.originGeometry = originGeometry;
}

@Override
public void setAttributes(GridElement el, Map<String, Object> attributes) {
attributes.put("id", ++id);
}

/**
* 过滤条件
*
* @param el
* @return
*/
@Override
public boolean getCreateFeature(GridElement el) {
/**
* intersects两种情况满足其一即为true
* 1. 相交
* 2. 包含
*/
return originGeometry.intersects(el.toGeometry());
}
}

实现一个网格Creator

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
import org.geotools.data.simple.SimpleFeatureIterator;
import org.geotools.data.simple.SimpleFeatureSource;
import org.geotools.geometry.jts.ReferencedEnvelope;
import org.geotools.grid.GridFeatureBuilder;
import org.geotools.grid.Grids;
import org.geotools.referencing.crs.DefaultGeographicCRS;
import org.locationtech.jts.geom.*;
import org.locationtech.jts.operation.buffer.BufferOp;
import org.opengis.feature.simple.SimpleFeature;
import org.opengis.referencing.crs.CoordinateReferenceSystem;

import java.util.LinkedList;
import java.util.List;

/**
* 网格创建工具类
* 默认使用84坐标系
*
* @author chenchuancheng github.com/meethigher
* @since 2022/5/15 21:36
*/
public class GridCreator {

/**
* 坐标系
* 默认使用84坐标系
*/
private CoordinateReferenceSystem crs;

/**
* 构建器
*/
private GridFeatureBuilder gridFeatureBuilder;


/**
* 边长
*/
private Double sideLen;

/**
* 区域
*/
private Geometry geometry;


public GridCreator(Double sideLen, Geometry geometry) {
this(DefaultGeographicCRS.WGS84, null, sideLen, geometry);
}

public GridCreator(CoordinateReferenceSystem crs, Double sideLen, Geometry geometry) {
this(crs, null, sideLen, geometry);
}


public GridCreator(GridFeatureBuilder gridFeatureBuilder, Double sideLen, Geometry geometry) {
this(DefaultGeographicCRS.WGS84, gridFeatureBuilder, sideLen, geometry);
}

public GridCreator(CoordinateReferenceSystem crs, GridFeatureBuilder gridFeatureBuilder, Double sideLen, Geometry geometry) {
this.crs = crs;
this.gridFeatureBuilder = gridFeatureBuilder;
this.sideLen = sideLen;
this.geometry = geometry;
}

/**
* LineSting拆分方格逻辑
*
* @param geometry
* @return
*/
private List<String> linestringSplit(Geometry geometry) {
List<String> list = new LinkedList<>();
Coordinate[] coordinates = geometry.getCoordinates();
for (int i = 0; i < coordinates.length; i++) {
if ((i + 1) > geometry.getCoordinates().length - 1) {
break;
} else {
Coordinate start = coordinates[i];
Coordinate end = coordinates[i + 1];
double deltaY = (end.y - start.y);
double deltaX = (end.x - start.x);
//加该判断是为了针对奇葩问题。比如手一抖,连续点了好几个一样的点。
//至于特殊情况,像π、π/2,通过Math.atan取绝对值就够了
if (!(deltaX == 0 && deltaY == 0)) {
double degree = Math.atan(deltaY / deltaX);
//如果距离不足sideLen时,至少保证有一个点被渲染
Geometry grid = GeoUtils.rotateGeometry(GeoUtils.createPoint(start.getX(), start.getY()).buffer(sideLen / 2, 1, BufferOp.CAP_SQUARE),
GeoUtils.createPoint(start.getX(), start.getY()), degree, false);
list.add(grid.toText());
//此处选用1/2的原因,看图。只要<俺>到<终点>的距离大于边长的一半,就给再补一个块出来
// ____________
//| __|__俺__|_终点
//|_____|_____|
while (GeoUtils.createLine(start, end).getLength() > sideLen * 1 / 2) {
double x = sideLen * Math.cos(degree);
double y = sideLen * Math.sin(degree);
double realX;
double realY;
/*如果线的走向是往西,需要取相反数*/
if (end.x >= start.x) {
realX = (start.x + x);
realY = (start.y + y);
} else {
realX = (start.x - x);
realY = (start.y - y);
}
Point point = GeoUtils.createPoint(realX, realY);
grid = GeoUtils.rotateGeometry(GeoUtils.createPoint(point.getX(), point.getY()).buffer(sideLen / 2, 1, BufferOp.CAP_SQUARE),
GeoUtils.createPoint(point.getX(), point.getY()), degree, false);
list.add(grid.toText());
start = new Coordinate(point.getX(), point.getY());

}
}
}
}
return list;
}

/**
* Polygon拆分方格逻辑
*
* @param geometry
* @return
*/
private List<String> polygonSplit(Geometry geometry) {
//以东西为x轴,南北为y轴,获取包含此几何图形中最小和最大x和y值
//如果是一条斜线,就重组x、y坐标构成一个矩形。
Envelope envelopeInternal = geometry.getEnvelopeInternal();
double maxX = envelopeInternal.getMaxX();
double minX = envelopeInternal.getMinX();
double maxY = envelopeInternal.getMaxY();
double minY = envelopeInternal.getMinY();

/**
* 多加一块的原因有两个
* 1. 如果sideLen超过了最大的差值,此时会缺块
* 2. 如果区域多出来了一部分,不足以用sideLen进行分块,此时也会缺块
*/
if ((maxX - minX) < sideLen || (maxX - minX) % sideLen < sideLen) {
maxX += sideLen;
}
if ((maxY - minY) < sideLen || (maxY - minY) % sideLen < sideLen) {
maxY += sideLen;
}

try {
//构建边界,使用84坐标系即可。
ReferencedEnvelope gridBounds =
new ReferencedEnvelope(minX, maxX, minY, maxY, DefaultGeographicCRS.WGS84);
SimpleFeatureSource grid;
if (gridFeatureBuilder == null) {
grid = Grids.createSquareGrid(gridBounds, sideLen);
} else {
grid = Grids.createSquareGrid(gridBounds, sideLen, -1, gridFeatureBuilder);
}
List<String> gridList = new LinkedList<>();
SimpleFeatureIterator iterator = grid.getFeatures().features();
while (iterator.hasNext()) {
SimpleFeature feature = iterator.next();
Geometry defaultGeometry = (Geometry) feature.getDefaultGeometry();
gridList.add(defaultGeometry.toString());
}
return gridList;
} catch (Exception e) {
return null;
}
}

/**
* 创建方格
*
* @return 方格的wkt集合
*/
public List<String> create() {
if (geometry == null) {
return null;
}
if (geometry instanceof LineString) {
return linestringSplit(geometry);
} else {
return polygonSplit(geometry);
}
}


}

进行测试

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
/**
* 创建块及自身,按米来切分,最后转成度
* 获取结果方便地图展示
* 地图https://meethigher.top/wkt
*
* @throws Exception
*/
@Test
public void getResultShowOnEarth() throws Exception {
String wkt = "LINESTRING(106.67105340852869 29.530636809322942,106.66803216986591 29.52328796943324)";
//String wkt = "LINESTRING(106.66641855239867 29.52517003980182,106.67002344131468 29.532727843679083,106.67507028736871 29.530487446637224,106.67675256676739 29.533175917132013)";
//String wkt = "POLYGON((106.65735483169556 29.530158851923517,106.65258264620205 29.52902369502023,106.64671182632448 29.530965403423963,106.64609384562937 29.52785865163041,106.64588785250089 29.524781676802462,106.65330362477108 29.522899600833085,106.6530289655202 29.524542686760086,106.65141534831493 29.527111822862878,106.6533379565226 29.527380681416773,106.65756082482405 29.526514355331983,106.65776682057185 29.52403482645859,106.66192102537026 29.522122860471057,106.67208337731424 29.521585112423097,106.67551660747264 29.522929476044652,106.67881250276693 29.52531940968403,106.67943048477169 29.528814585054363,106.68025446252427 29.532459000021845,106.67874383926389 29.53428115740634,106.6783661873778 29.532548613742065,106.67850351438389 29.530218596454873,106.6765122413635 29.531114763609096,106.67575693235264 29.5339824452562,106.67469262972004 29.536192894729282,106.67187738418576 29.53732797116615,106.67012643866471 29.53583444659172,106.67246103443901 29.5338330889642,106.67290735244748 29.531831692071208,106.66882181115214 29.53189143550047,106.66703653387954 29.53278758783246,106.66264200158184 29.535655222388314,106.65986108779906 29.536909786679814,106.65821314015191 29.5349084499357,106.65838480152888 29.532309639998616,106.65735483169556 29.530158851923517))";

Geometry epsg3857 = GeoUtils.toEPSG3857(GeoUtils.wktToGeo(wkt));
GridCreator gridCreator = new GridCreator(100.0, epsg3857);
log.info("start");
List<String> list = gridCreator.create();
log.info("end-{}", list.size());
List<String> newList = new LinkedList<>();
newList.add(wkt);
Iterator<String> iterator = list.iterator();
while (iterator.hasNext()) {
newList.add(GeoUtils.toEPSG4326(GeoUtils.wktToGeo(iterator.next())).toText());
iterator.remove();
}
System.out.println(newList.toString().replaceAll("\\[|\\]", ""));
}

最后结果展示如图,符合需求了。

总体上,没什么特别复杂的,就是调用geotools提供的api就可以了。

最大的收获,还是发现了这些大佬写的wkt展示的页面吧,页面有点难用,有时间再优化一下,现在先将就用着。

三、小插曲

在实现LineString线的拆分过程中,又重拾了还给老师的三角函数。

这边一个难点是,已知点A和中心点P和degree,求旋转degree后的B的坐标。

思路:

  1. A相对于P,就相当于平移后的a相当于原点o。a与x轴夹角可知为θ,旋转半径可知为r。
  2. A旋转degree后到B,就相当于a旋转degree后到b。
  3. a旋转后得到的b与x轴夹角为θ-degree,由于旋转半径已知,可求出b
  4. b平移得到最终B

关键逻辑代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
/**
* 旋转
* 在平面坐标系算会比较准,建议旋转时使用3857坐标系
* <p>
* 数学题,A以P为中心,求旋转degree后B的坐标
*
* @param P 旋转中心点
* @param A 待旋转的点
* @param degree 度数
* @param isClockwise true为顺时针,false为逆时针
* @return A以P为中心旋转degree后B的坐标
*/
public static Point rotatePoint(Point A, Point P,
double degree, boolean isClockwise) {
//将点A平移到点a。点A以P为中心<==>点a以原点o为中心
Point a = GeoUtils.createPoint(A.getX() - P.getX(), A.getY() - P.getY());
//以原点o为中心的圆半径<==>以P为中心的圆的半径
double radius = pointLength(a, GeoUtils.createPoint(0.0, 0.0));
//线段oa与x轴的角度
double degreeWithX = degree(a.getX(), a.getY());
//线段oa旋转后与x轴的角度
double degreeAfterRotateWithX = degreeWithX - (isClockwise ? 1 : -1) * degree;
//获取到旋转后的点b
Point b = GeoUtils.createPoint(radius * Math.cos(degreeAfterRotateWithX),
radius * Math.sin(degreeAfterRotateWithX));
//平移获取最终B点
return GeoUtils.createPoint(b.getX() + P.getX(), b.getY() + P.getY());
}
发布:2022-05-15 02:44:39
修改:2023-01-17 22:56:27
链接:https://meethigher.top/blog/2022/geometry-grid/
标签:java gis 
付款码 打赏 分享
若无法评论请科学上网
Shift+Ctrl+1 可控制工具栏