Geo Tools Vector grids 矢量栅格
矢量栅格化
GeoTools 矢量网格类使创建由多边形或线元素组成的矢量网格(也称为晶格)变得容易,每个网格都表示为SimpleFeature。
使用“网格”或“线”实用程序类可以轻松生成简单的网格,而当需要对网格布局和属性进行更多控制时,可以使用较低级别的类。

<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>
<repository>
<id>osgeo-snapshot</id>
<name>OSGeo Snapshot Repository</name>
<url>https://repo.osgeo.org/repository/snapshot/</url>
<snapshots><enabled>true</enabled></snapshots>
<releases><enabled>false</enabled></releases>
</repository>
</repositories>
<dependencies>
<dependency>
<groupId>org.geotools</groupId>
<artifactId>gt-grid</artifactId>
<version>${geotools.version}</version>
</dependency>
</dependencies>
/**
* 获取栅格化后的 BoundingBox 集合
*
* @param srcGeometry 栅格前的 geometry
* @param srcBoundingBox 栅格前的 boundingBox
* @return 栅格化后的 BoundingBox 集合
* @throws Exception 异常
*/
private List<BoundingBox> gridGeometry(Geometry srcGeometry, BoundingBox srcBoundingBox) throws Exception {
//获取栅格精度 (单位 度 °)
double sideLen = getSideLen(srcBoundingBox);
//获取 envelope
ReferencedEnvelope envelope = getReferencedEnvelope(srcBoundingBox, sideLen);
//栅格化
SimpleFeatureSource grid = Grids.createSquareGrid(envelope, sideLen);
SimpleFeatureCollection features = grid.getFeatures();
SimpleFeatureIterator iterator = features.features();
List<BoundingBox> boundingBoxes = new LinkedList<>();
while (iterator.hasNext()) {
SimpleFeature feature = iterator.next();
Geometry geometry = (Geometry) feature.getDefaultGeometry();
//相交计算
if (srcGeometry.intersects(geometry)) {
BoundingBox bounds = feature.getBounds();
boundingBoxes.add(bounds);
}
}
return boundingBoxes;
}
//================================================= Gird ========================================================
/**
* 获取栅格化的 Envelope
*
* @param boundingBox 原 boundingBox
* @param sideLen 栅格精度
* @return Envelope
*/
private ReferencedEnvelope getReferencedEnvelope(BoundingBox boundingBox, double sideLen) {
double maxX = getDoubleValueUp(boundingBox.getMaxX(), sideLen);
double maxY = getDoubleValueUp(boundingBox.getMaxY(), sideLen);
double minX = getDoubleValueDown(boundingBox.getMinX(), sideLen);
double minY = getDoubleValueDown(boundingBox.getMinY(), sideLen);
return new ReferencedEnvelope(minX, maxX, minY, maxY, DefaultGeographicCRS.WGS84);
}
/**
* 获取栅格精度值
*
* @param boundingBox box
* @return 栅格精度 以 度 为单位
*/
private double getSideLen(BoundingBox boundingBox) {
double maxX = boundingBox.getMaxX();
double maxY = boundingBox.getMaxY();
double minX = boundingBox.getMinX();
double minY = boundingBox.getMinY();
double xBetween = Math.abs(new BigDecimal(Double.toString(maxX)).subtract(new BigDecimal(Double.toString(minX))).doubleValue());
double yBetween = Math.abs(new BigDecimal(Double.toString(maxY)).subtract(new BigDecimal(Double.toString(minY))).doubleValue());
// SIDE_LEN_SIZE = 边长格网数 可根据实际情况自定义
double SIDE_LEN_SIZE = 50.0;
double result = new BigDecimal(Double.toString(Math.min(xBetween, yBetween))).divide(new BigDecimal(SIDE_LEN_SIZE)).doubleValue();
result = Math.max(result, 0.01);
return Double.parseDouble(String.format("%.4f", result));
}
/**
* 向上取值
*
* @param value 值
* @param sideLen 栅格精度
* @return 值
*/
private double getDoubleValueUp(double value, double sideLen) {
NumberFormat nf = NumberFormat.getNumberInstance();
nf.setMaximumFractionDigits(2);
nf.setRoundingMode(RoundingMode.UP);
String format = nf.format(value);
return Math.min(Double.parseDouble(format) + sideLen, 180);
}
/**
* 向下取值
*
* @param value 值
* @param sideLen 栅格精度
* @return 值
*/
private double getDoubleValueDown(double value, double sideLen) {
NumberFormat nf = NumberFormat.getNumberInstance();
nf.setMaximumFractionDigits(2);
nf.setRoundingMode(RoundingMode.DOWN);
String format = nf.format(value);
return Math.max(Double.parseDouble(format) - sideLen, -180);
}
效果图
geojson.io
Mr.superbeyone