Geotools拆分栅格数据然后再合并它不起作用。

huangapple go评论140阅读模式
英文:

Geotools splitting raster and merging it again doesn't work

问题

以下是代码部分的中文翻译:

  1. public ArrayList<GridCoverage2D> createTiles(GridCoverage2D grid) throws FactoryException, TransformException {
  2. // 代码内容...
  3. }

请注意,这只是代码的翻译,不包括代码之外的任何内容。如果您需要更多帮助或有其他问题,请随时提出。

英文:

first time posting a question here so I'm sorry for any error.

We are trying to let the user choose a raster image (.tiff) so that it can be uploaded to hbase for processing and other operations. Since single images can be quite large, we are trying to split them so that the upload process can be hasten, and then we merge them together again to get the original single image.<br><br>From what I understood, the error seems to lie in the split method.
At the moment we are just trying to split the image and rebuild it locally.

This is the code for the split method

  1. public ArrayList&lt;GridCoverage2D&gt; createTiles(GridCoverage2D grid) throws FactoryException, TransformException {
  2. Geometry geom = buildGeometryFromGrid(grid);
  3. org.locationtech.jts.geom.Envelope env = geom.getEnvelopeInternal();
  4. Coverage coverage = GeoHash.coverBoundingBoxMaxHashes(env.getMaxY(), env.getMinX(), env.getMinY(), env.getMaxX(), 1);
  5. ArrayList&lt;String&gt; geohashes = new ArrayList&lt;&gt;();
  6. String hash = null;
  7. if (coverage == null) hash = &quot;&quot;;
  8. else {
  9. for (String singleHash : coverage.getHashes()) {
  10. hash = singleHash;
  11. break;
  12. }
  13. }
  14. if (hash.length() == Constants.defaultPrecision) {
  15. coverage = GeoHash.coverBoundingBox(env.getMaxY(), env.getMinX(), env.getMinY(), env.getMaxX(), Constants.defaultPrecision + 1);
  16. geohashes = new ArrayList&lt;&gt;(coverage.getHashes());
  17. }
  18. else if (hash.length() &lt; Constants.defaultPrecision) {
  19. coverage = GeoHash.coverBoundingBox(env.getMaxY(), env.getMinX(), env.getMinY(), env.getMaxX(), Constants.defaultPrecision);
  20. geohashes = new ArrayList&lt;&gt;(coverage.getHashes());
  21. }
  22. else {
  23. if (hash.length() &gt; Constants.spaceLength) geohashes.add(hash.substring(0, hash.length() - 1));
  24. else if (hash.length() == Constants.spaceLength) geohashes.add(hash);
  25. else {
  26. coverage = GeoHash.coverBoundingBox(env.getMaxY(), env.getMinX(), env.getMinY(), env.getMaxX(), hash.length() + 1);
  27. geohashes = new ArrayList&lt;&gt;(coverage.getHashes());
  28. }
  29. }
  30. System.out.println(geohashes.size());
  31. List&lt;ReferencedEnvelope&gt; envelopes = calcuateCuts(geohashes);
  32. ArrayList&lt;GridCoverage2D&gt; tiles = new ArrayList&lt;&gt;();
  33. for (ReferencedEnvelope envelope : envelopes) {
  34. tiles.addAll(getTiles(envelope, grid));
  35. }
  36. System.out.println(tiles.size());
  37. return tiles;
  38. }

It kinda works because we get a number of tiffs, but we also get some of them that are not being visualized with tools like qgis. The merge method gives us back a tiff image, but then again it is not visualized in qgis.

Tell me if you need more informations.
Thanks!

EDIT

This is the gdalinfo for the original and correct tif

  1. Driver: GTiff/GeoTIFF
  2. Files: C:\Users\Consulente\Desktop\appoggio_rilascio\originale.tif
  3. C:\Users\Consulente\Desktop\appoggio_rilascio\originale.tif.aux.xml
  4. Size is 266, 269
  5. Coordinate System is:
  6. PROJCRS[&quot;WGS 84 / UTM zone 33N&quot;,
  7. BASEGEOGCRS[&quot;WGS 84&quot;,
  8. DATUM[&quot;World Geodetic System 1984&quot;,
  9. ELLIPSOID[&quot;WGS 84&quot;,6378137,298.257223563,
  10. LENGTHUNIT[&quot;metre&quot;,1]]],
  11. PRIMEM[&quot;Greenwich&quot;,0,
  12. ANGLEUNIT[&quot;degree&quot;,0.0174532925199433]],
  13. ID[&quot;EPSG&quot;,4326]],
  14. CONVERSION[&quot;UTM zone 33N&quot;,
  15. METHOD[&quot;Transverse Mercator&quot;,
  16. ID[&quot;EPSG&quot;,9807]],
  17. PARAMETER[&quot;Latitude of natural origin&quot;,0,
  18. ANGLEUNIT[&quot;degree&quot;,0.0174532925199433],
  19. ID[&quot;EPSG&quot;,8801]],
  20. PARAMETER[&quot;Longitude of natural origin&quot;,15,
  21. ANGLEUNIT[&quot;degree&quot;,0.0174532925199433],
  22. ID[&quot;EPSG&quot;,8802]],
  23. PARAMETER[&quot;Scale factor at natural origin&quot;,0.9996,
  24. SCALEUNIT[&quot;unity&quot;,1],
  25. ID[&quot;EPSG&quot;,8805]],
  26. PARAMETER[&quot;False easting&quot;,500000,
  27. LENGTHUNIT[&quot;metre&quot;,1],
  28. ID[&quot;EPSG&quot;,8806]],
  29. PARAMETER[&quot;False northing&quot;,0,
  30. LENGTHUNIT[&quot;metre&quot;,1],
  31. ID[&quot;EPSG&quot;,8807]]],
  32. CS[Cartesian,2],
  33. AXIS[&quot;(E)&quot;,east,
  34. ORDER[1],
  35. LENGTHUNIT[&quot;metre&quot;,1]],
  36. AXIS[&quot;(N)&quot;,north,
  37. ORDER[2],
  38. LENGTHUNIT[&quot;metre&quot;,1]],
  39. USAGE[
  40. SCOPE[&quot;unknown&quot;],
  41. AREA[&quot;World - N hemisphere - 12┬░E to 18┬░E - by country&quot;],
  42. BBOX[0,12,84,18]],
  43. ID[&quot;EPSG&quot;,32633]]
  44. Data axis to CRS axis mapping: 1,2
  45. Origin = (293436.000000000000000,4647354.000000000000000)
  46. Pixel Size = (3.000000000000000,-3.000000000000000)
  47. Metadata:
  48. AREA_OR_POINT=Area
  49. Image Structure Metadata:
  50. COMPRESSION=LZW
  51. INTERLEAVE=PIXEL
  52. Corner Coordinates:
  53. Upper Left ( 293436.000, 4647354.000) ( 12d30&#39;28.04&quot;E, 41d57&#39; 4.05&quot;N)
  54. Lower Left ( 293436.000, 4646547.000) ( 12d30&#39;29.05&quot;E, 41d56&#39;37.91&quot;N)
  55. Upper Right ( 294234.000, 4647354.000) ( 12d31&#39; 2.66&quot;E, 41d57&#39; 4.80&quot;N)
  56. Lower Right ( 294234.000, 4646547.000) ( 12d31&#39; 3.68&quot;E, 41d56&#39;38.66&quot;N)
  57. Center ( 293835.000, 4646950.500) ( 12d30&#39;45.86&quot;E, 41d56&#39;51.36&quot;N)
  58. Band 1 Block=266x3 Type=UInt16, ColorInterp=Red
  59. Min=4230.000 Max=18752.000
  60. Minimum=4230.000, Maximum=18752.000, Mean=6493.946, StdDev=1353.641
  61. NoData Value=0
  62. Metadata:
  63. STATISTICS_MAXIMUM=18752
  64. STATISTICS_MEAN=6493.9456647057
  65. STATISTICS_MINIMUM=4230
  66. STATISTICS_STDDEV=1353.6413052408
  67. STATISTICS_VALID_PERCENT=94.55
  68. Band 2 Block=266x3 Type=UInt16, ColorInterp=Green
  69. Min=3979.000 Max=17611.000
  70. Minimum=3979.000, Maximum=17611.000, Mean=5971.019, StdDev=1344.279
  71. NoData Value=0
  72. Metadata:
  73. STATISTICS_MAXIMUM=17611
  74. STATISTICS_MEAN=5971.0194223549
  75. STATISTICS_MINIMUM=3979
  76. STATISTICS_STDDEV=1344.2792863221
  77. STATISTICS_VALID_PERCENT=94.55
  78. Band 3 Block=266x3 Type=UInt16, ColorInterp=Blue
  79. Min=2619.000 Max=16659.000
  80. Minimum=2619.000, Maximum=16659.000, Mean=4982.758, StdDev=1525.882
  81. NoData Value=0
  82. Metadata:
  83. STATISTICS_MAXIMUM=16659
  84. STATISTICS_MEAN=4982.7577379017
  85. STATISTICS_MINIMUM=2619
  86. STATISTICS_STDDEV=1525.8817457971
  87. STATISTICS_VALID_PERCENT=94.55
  88. Band 4 Block=266x3 Type=UInt16, ColorInterp=Undefined
  89. NoData Value=0

and this is the gdalinfo from the uncorrect merged tiff (which should have been equal to the original one)

  1. Driver: GTiff/GeoTIFF
  2. Files: C:\Users\Consulente\Desktop\appoggio_rilascio\merged_tif\merged.tiff
  3. Size is 266, 269
  4. Coordinate System is:
  5. GEOGCRS[&quot;WGS 84&quot;,
  6. DATUM[&quot;World Geodetic System 1984&quot;,
  7. ELLIPSOID[&quot;WGS 84&quot;,6378137,298.257223563,
  8. LENGTHUNIT[&quot;metre&quot;,1]]],
  9. PRIMEM[&quot;Greenwich&quot;,0,
  10. ANGLEUNIT[&quot;degree&quot;,0.0174532925199433]],
  11. CS[ellipsoidal,2],
  12. AXIS[&quot;geodetic latitude (Lat)&quot;,north,
  13. ORDER[1],
  14. ANGLEUNIT[&quot;degree&quot;,0.0174532925199433]],
  15. AXIS[&quot;geodetic longitude (Lon)&quot;,east,
  16. ORDER[2],
  17. ANGLEUNIT[&quot;degree&quot;,0.0174532925199433]],
  18. ID[&quot;EPSG&quot;,4326]]
  19. Data axis to CRS axis mapping: 2,1
  20. GeoTransform =
  21. 41.95133415197179, 0, -2.77698308382438e-05
  22. 12.50778763370388, 3.722213354286106e-05, 0
  23. Metadata:
  24. AREA_OR_POINT=Area
  25. TIFFTAG_RESOLUTIONUNIT=1 (unitless)
  26. TIFFTAG_XRESOLUTION=1
  27. TIFFTAG_YRESOLUTION=1
  28. Image Structure Metadata:
  29. INTERLEAVE=PIXEL
  30. Corner Coordinates:
  31. Upper Left ( 41.9513342, 12.5077876) ( 41d57&#39; 4.80&quot;E, 12d30&#39;28.04&quot;N)
  32. Lower Left ( 41.9438641, 12.5077876) ( 41d56&#39;37.91&quot;E, 12d30&#39;28.04&quot;N)
  33. Upper Right ( 41.9513342, 12.5176887) ( 41d57&#39; 4.80&quot;E, 12d31&#39; 3.68&quot;N)
  34. Lower Right ( 41.9438641, 12.5176887) ( 41d56&#39;37.91&quot;E, 12d31&#39; 3.68&quot;N)
  35. Center ( 41.9475991, 12.5127382) ( 41d56&#39;51.36&quot;E, 12d30&#39;45.86&quot;N)
  36. Band 1 Block=266x8 Type=UInt16, ColorInterp=Red
  37. NoData Value=0
  38. Band 2 Block=266x8 Type=UInt16, ColorInterp=Green
  39. NoData Value=0
  40. Band 3 Block=266x8 Type=UInt16, ColorInterp=Blue
  41. NoData Value=0
  42. Band 4 Block=266x8 Type=UInt16, ColorInterp=Alpha
  43. NoData Value=0

答案1

得分: 1

从查看GdalInfo输出,您已将CRS更改为EPSG:4326并颠倒了纬度/经度值 - 因此,在您发布的代码中可能存在某种方式来写出瓦片的问题。

如果我要尝试这样做,我会查看图像切片教程,其中关键的内部循环如下:

  1. // 遍历我们的瓦片计数
  2. for (int i = 0; i < htc; i++) {
  3. for (int j = 0; j < vtc; j++) {
  4. System.out.println("Processing tile at indices i: " + i + " and j: " + j);
  5. // 创建瓦片的包围框
  6. Envelope envelope = getTileEnvelope(coverageMinX, coverageMinY, geographicTileWidth, geographicTileHeight,
  7. targetCRS, i, j);
  8. GridCoverage2D finalCoverage = cropCoverage(gridCoverage, envelope);
  9. if (this.getTileScale() != null) {
  10. finalCoverage = scaleCoverage(finalCoverage);
  11. }
  12. // 使用AbstractGridFormat的写入器来写出瓦片
  13. fileExtension = "png";
  14. File tileFile = new File(tileDirectory, i + "_" + j + "." + fileExtension);
  15. final WorldImageWriter wiWriter = new WorldImageWriter(tileFile);
  16. // 写入PNG的参数
  17. final Format oFormat = wiWriter.getFormat();
  18. ((AbstractGridFormat) oFormat).getWriter(tileFile).write(finalCoverage, null);
  19. }
  20. }

以及使用CoverageProcessor类实现的cropCoverage方法:

  1. private GridCoverage2D cropCoverage(GridCoverage2D gridCoverage, Envelope envelope) {
  2. CoverageProcessor processor = CoverageProcessor.getInstance();
  3. // 手动创建我们想要的操作和参数的示例
  4. final ParameterValueGroup param = processor.getOperation("CoverageCrop").getParameters();
  5. param.parameter("Source").setValue(gridCoverage);
  6. param.parameter("Envelope").setValue(envelope);
  7. return (GridCoverage2D) processor.doOperation(param);
  8. }

请注意,这些代码片段是Java代码,用于处理地理信息数据的图像切片。

英文:

From looking at the GdalInfo output you have changed the CRS to EPSG:4326 and flipped the lat/lon values - so something is going on in the way you write out the tiles which you haven't posted the code for.

If I was trying to do this I would look at the Image Tiling tutorial, the key inner loop is this:

  1. // iterate over our tile counts
  2. for (int i = 0; i &lt; htc; i++) {
  3. for (int j = 0; j &lt; vtc; j++) {
  4. System.out.println(&quot;Processing tile at indices i: &quot; + i + &quot; and j: &quot; + j);
  5. // create the envelope of the tile
  6. Envelope envelope = getTileEnvelope(coverageMinX, coverageMinY, geographicTileWidth, geographicTileHeight,
  7. targetCRS, i, j);
  8. GridCoverage2D finalCoverage = cropCoverage(gridCoverage, envelope);
  9. if (this.getTileScale() != null) {
  10. finalCoverage = scaleCoverage(finalCoverage);
  11. }
  12. // use the AbstractGridFormat&#39;s writer to write out the tile
  13. fileExtension = &quot;png&quot;;
  14. File tileFile = new File(tileDirectory, i + &quot;_&quot; + j + &quot;.&quot; + fileExtension);
  15. final WorldImageWriter wiWriter = new WorldImageWriter(tileFile);
  16. // writing parameters for png
  17. final Format oFormat = wiWriter.getFormat();
  18. ((AbstractGridFormat) oFormat).getWriter(tileFile).write(finalCoverage, null);
  19. }
  20. }

and cropCoverage implemented using the CoverageProcessor class:

  1. private GridCoverage2D cropCoverage(GridCoverage2D gridCoverage, Envelope envelope) {
  2. CoverageProcessor processor = CoverageProcessor.getInstance();
  3. // An example of manually creating the operation and parameters we want
  4. final ParameterValueGroup param = processor.getOperation(&quot;CoverageCrop&quot;).getParameters();
  5. param.parameter(&quot;Source&quot;).setValue(gridCoverage);
  6. param.parameter(&quot;Envelope&quot;).setValue(envelope);
  7. return (GridCoverage2D) processor.doOperation(param);
  8. }

huangapple
  • 本文由 发表于 2020年8月11日 23:15:31
  • 转载请务必保留本文链接:https://go.coder-hub.com/63361200.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定