GeoCastUtil.java 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331
  1. package com.flyer.util;
  2. import org.apache.commons.lang3.ObjectUtils;
  3. import org.geotools.geojson.geom.GeometryJSON;
  4. import org.geotools.geometry.jts.JTS;
  5. import org.geotools.referencing.CRS;
  6. import org.geotools.referencing.GeodeticCalculator;
  7. import org.geotools.referencing.crs.DefaultGeographicCRS;
  8. import org.locationtech.jts.geom.*;
  9. import org.locationtech.jts.io.ParseException;
  10. import org.locationtech.jts.io.WKTReader;
  11. import org.locationtech.jts.io.WKTWriter;
  12. import org.opengis.referencing.FactoryException;
  13. import org.opengis.referencing.crs.CoordinateReferenceSystem;
  14. import org.opengis.referencing.operation.MathTransform;
  15. import org.opengis.referencing.operation.TransformException;
  16. import java.io.IOException;
  17. import java.io.StringReader;
  18. import java.util.List;
  19. public class GeoCastUtil {
  20. public static GeometryJSON gjson = new GeometryJSON(20);
  21. private static WKTWriter wKTWriter2 = new WKTWriter();
  22. private static WKTReader wktReader2 = new WKTReader();
  23. public static String geomToWkt(Geometry geometry){
  24. return wKTWriter2.write(geometry);
  25. }
  26. public static Geometry wktToGeom(String wkt){
  27. try {
  28. return wktReader2.read(wkt);
  29. } catch (ParseException e) {
  30. e.printStackTrace();
  31. }
  32. return null;
  33. }
  34. /**
  35. * 创建矩形wkt
  36. * @param points
  37. * @return
  38. */
  39. public static String createPolygonWkt(List<double[]> points){
  40. if(ObjectUtils.isEmpty(points)){
  41. return "";
  42. }
  43. StringBuffer sb = new StringBuffer(points.size() * 50);
  44. sb.append("MULTIPOLYGON(((");
  45. for(double[] point : points){
  46. sb.append(point[0]);
  47. sb.append(" ");
  48. sb.append(point[1]);
  49. sb.append(",");
  50. }
  51. sb.append(points.get(0)[0]);
  52. sb.append(" ");
  53. sb.append(points.get(0)[1]);
  54. sb.append(")))");
  55. return sb.toString();
  56. }
  57. private static GeometryCollection castGeometryCollection2(GeometryJSON fjson, GeometryCollection geometryCollection) throws IOException {
  58. if (geometryCollection.getGeometryType().equals("GeometryCollection")) {
  59. return geometryCollection;
  60. }
  61. String[] geometriesJsonArray = new String[geometryCollection.getNumGeometries()];
  62. int length = 0;
  63. for(int i=0; i< geometryCollection.getNumGeometries(); i++){
  64. Geometry geometry = geometryCollection.getGeometryN(i);
  65. geometriesJsonArray[i] = fjson.toString(geometry);
  66. length += geometriesJsonArray[i].length();
  67. }
  68. StringBuffer result = new StringBuffer(length + 100);
  69. result.append("{\"type\": \"GeometryCollection\",\"geometries\": [");
  70. result.append(geometriesJsonArray[0]);
  71. for(int i=1;i< geometriesJsonArray.length;i++){
  72. result.append(",").append(geometriesJsonArray[i]);
  73. }
  74. result.append("]}");
  75. return fjson.readGeometryCollection(new StringReader(result.toString()));
  76. }
  77. public static GeometryCollection castGeometryCollection(GeometryJSON fjson, Geometry geometry) throws IOException {
  78. if (geometry.getGeometryType().equals("GeometryCollection")) {
  79. return (GeometryCollection)geometry;
  80. }
  81. if (geometry instanceof GeometryCollection) {
  82. return castGeometryCollection2(fjson, (GeometryCollection)geometry);
  83. }
  84. String geometryJson = fjson.toString(geometry);
  85. StringBuffer result = new StringBuffer(geometryJson.length() + 100);
  86. result.append("{\"type\": \"GeometryCollection\",\"geometries\": [");
  87. result.append(geometryJson);
  88. result.append("]}");
  89. return fjson.readGeometryCollection(new StringReader(result.toString()));
  90. }
  91. public static Geometry readGeometry(Object input, String type, GeometryJSON gjson) throws IOException {
  92. switch(type){
  93. case "Point":
  94. return gjson.readPoint(input);
  95. case "MultiPoint":
  96. return gjson.readMultiPoint(input);
  97. case "LineString":
  98. return gjson.readLine(input);
  99. case "MultiLineString":
  100. return gjson.readMultiLine(input);
  101. case "Polygon":
  102. return gjson.readPolygon(input);
  103. case "MultiPolygon":
  104. return gjson.readMultiPolygon(input);
  105. }
  106. return null;
  107. }
  108. public static Double[] lonlatToMactor(Double[] point){
  109. CoordinateReferenceSystem crsTarget = null;
  110. // 投影转换
  111. MathTransform transform = null;
  112. try {
  113. crsTarget = CRS.decode("EPSG:3857");
  114. transform = CRS.findMathTransform(DefaultGeographicCRS.WGS84, crsTarget);
  115. } catch (FactoryException e) {
  116. e.printStackTrace();
  117. }
  118. Coordinate sourcePoint = new Coordinate(point[0],point[1]);
  119. Coordinate targetPoint = new Coordinate();
  120. try {
  121. JTS.transform(sourcePoint, targetPoint, transform);
  122. } catch (TransformException e) {
  123. e.printStackTrace();
  124. }
  125. return new Double[]{targetPoint.getX(), targetPoint.getY()};
  126. }
  127. public static double[] mactorToLonlat(double[] point, CoordinateReferenceSystem srcTarget){
  128. // 投影转换
  129. MathTransform transform = null;
  130. try {
  131. transform = CRS.findMathTransform(srcTarget, DefaultGeographicCRS.WGS84);
  132. } catch (FactoryException e) {
  133. e.printStackTrace();
  134. }
  135. Coordinate sourcePoint = new Coordinate(point[0],point[1]);
  136. Coordinate targetPoint = new Coordinate();
  137. try {
  138. JTS.transform(sourcePoint, targetPoint, transform);
  139. } catch (TransformException e) {
  140. e.printStackTrace();
  141. }
  142. return new double[]{targetPoint.getX(),targetPoint.getY()};
  143. }
  144. public static double[] mactorToLonlat(double[] point){
  145. CoordinateReferenceSystem srcTarget = null;
  146. // 投影转换
  147. MathTransform transform = null;
  148. try {
  149. srcTarget = CRS.decode("EPSG:3857");
  150. transform = CRS.findMathTransform(srcTarget, DefaultGeographicCRS.WGS84);
  151. } catch (FactoryException e) {
  152. e.printStackTrace();
  153. }
  154. Coordinate sourcePoint = new Coordinate(point[0],point[1]);
  155. Coordinate targetPoint = new Coordinate();
  156. try {
  157. JTS.transform(sourcePoint, targetPoint, transform);
  158. } catch (TransformException e) {
  159. e.printStackTrace();
  160. }
  161. return new double[]{targetPoint.getX(),targetPoint.getY()};
  162. }
  163. /**
  164. * 根据多边形类型计算出多边形面积,单位(平方米)
  165. * @param geometry 多边形对象
  166. * @return 面积
  167. */
  168. public static double getArea(Geometry geometry){
  169. CoordinateReferenceSystem source = null;
  170. try {
  171. source = CRS.decode("CRS:84");
  172. } catch (FactoryException e) {
  173. e.printStackTrace();
  174. }
  175. CoordinateReferenceSystem target = null;
  176. try {
  177. target = CRS.decode("EPSG:3857");
  178. } catch (FactoryException e) {
  179. e.printStackTrace();
  180. }
  181. MathTransform transform = null;
  182. try {
  183. transform = CRS.findMathTransform(source, target, true);
  184. } catch (FactoryException e) {
  185. e.printStackTrace();
  186. }
  187. Geometry transform1 = null;
  188. try {
  189. transform1 = JTS.transform(geometry, transform);
  190. } catch (TransformException e) {
  191. e.printStackTrace();
  192. }
  193. double area = transform1.getArea();
  194. return area;
  195. }
  196. // public static double toFlex(double d){
  197. // BigDecimal bd = new BigDecimal(d);
  198. // String s = bd.toPlainString();
  199. // int index = s.indexOf(".");
  200. // String last = s.substring(index + 1);
  201. // if(last.length() > 4){
  202. // last = last.substring(0,4);
  203. // }
  204. // return Double.parseDouble(s.substring(0,index + 1) + last);
  205. // }
  206. public static int dist84(Point p1,Point p2){
  207. // 84坐标系构造GeodeticCalculator
  208. GeodeticCalculator geodeticCalculator = new GeodeticCalculator(DefaultGeographicCRS.WGS84);
  209. // 起点经纬度
  210. geodeticCalculator.setStartingGeographicPoint(p1.getX(),p1.getY());
  211. // 末点经纬度
  212. geodeticCalculator.setDestinationGeographicPoint(p2.getX(),p2.getY());
  213. // 计算距离,单位:米
  214. double orthodromicDistance = geodeticCalculator.getOrthodromicDistance();
  215. return (int)orthodromicDistance;
  216. }
  217. public static double miToMu(double m){
  218. double c = 666.66666666667;
  219. return m / c;
  220. }
  221. public static double mToDegree(Integer m){
  222. double degree = m / (2 * Math.PI * 6371004) * 360;
  223. return degree;
  224. }
  225. /**
  226. * 返回面积 单位亩
  227. * @param geom
  228. * @return
  229. */
  230. public static double getAreaOfMu(Geometry geom){
  231. return miToMu(getArea(geom));
  232. }
  233. /**
  234. * 转为geom并合并
  235. * @param geom
  236. * @return
  237. */
  238. public static Geometry unionToGeom(Geometry... geom){
  239. if(geom.length == 0){
  240. return null;
  241. }
  242. if(geom.length == 1){
  243. return geom[0];
  244. }
  245. Geometry res = geom[0];
  246. for(int i=1;i<geom.length;i++){
  247. res = res.union(geom[i]);
  248. }
  249. return res;
  250. }
  251. /**
  252. * 转为geom并合并
  253. * @param wkt
  254. * @return
  255. */
  256. public static Geometry unionToGeom(String... wkt){
  257. if(wkt.length == 1){
  258. return wktToGeom(wkt[0]);
  259. }
  260. Geometry res = wktToGeom(wkt[0]);
  261. for(int i=1;i<wkt.length;i++){
  262. res = res.union(wktToGeom(wkt[i]));
  263. }
  264. return res;
  265. }
  266. public static double distance(Point p1, Point p2){
  267. double distance = p1.distance(p2);
  268. return distance;
  269. }
  270. private static GeometryFactory geometryFactory = new GeometryFactory();
  271. // 假设的参数113.746646,22.970229
  272. private static final double REFERENCE_LATITUDE = 22.970229; // 参考点的纬度
  273. private static final double REFERENCE_LONGITUDE = 113.746646; // 参考点的经度
  274. private static final double M_PER_PIXEL_X = 0.001; // 每个像素的米数(x方向)
  275. private static final double M_PER_PIXEL_Y = 0.001; // 每个像素的米数(y方向)
  276. public static Coordinate pixelToLatLng(double pixelX, double pixelY, double imageWidth, double imageHeight) {
  277. // 计算像素到米的转换
  278. double metersX = pixelX * M_PER_PIXEL_X;
  279. double metersY = pixelY * M_PER_PIXEL_Y;
  280. // 将米转换为经纬度
  281. // 假设地球是平坦的,这里只是简单的加减
  282. double latitude = REFERENCE_LATITUDE - (metersY / 111320); // 111320米为1度纬度距离的近似值
  283. double longitude = REFERENCE_LONGITUDE + (metersX / (111320 * Math.cos(Math.toRadians(REFERENCE_LATITUDE)))); // 1度经度距离随纬度变化
  284. return new Coordinate(longitude, latitude);
  285. }
  286. public static void main(String[] args) {
  287. int pixelX = 100;
  288. int pixelY = 100;
  289. int imageWidth = 1000;
  290. int imageHeight = 1000;
  291. Coordinate latLng = pixelToLatLng(pixelX, pixelY, imageWidth, imageHeight);
  292. System.out.println("Latitude: " + latLng.y + ", Longitude: " + latLng.x);
  293. }
  294. }