Geotrellis入门

一. 看起来很牛逼的几个链接

二. Geotrellis github

三. 使用

1. 建立工程

  按道理来讲,用sbt建立一个scala工程,并在build里引入依赖,就可以自动下载geotrellis的相关包,但我的sbt不知道怎么回事,依赖包下载不来,最后还是通过maven来解决的。
geotrellis的maven列表
  先用raster试试水,pom中加入

    <!-- https://mvnrepository.com/artifact/org.locationtech.geotrellis/geotrellis-spark -->
    <dependency>
      <groupId>org.locationtech.geotrellis</groupId>
      <artifactId>geotrellis-spark_2.11</artifactId>
      <version>2.1.0</version>
    </dependency>
    <!-- https://mvnrepository.com/artifact/org.locationtech.geotrellis/geotrellis-raster -->
    <dependency>
      <groupId>org.locationtech.geotrellis</groupId>
      <artifactId>geotrellis-raster_2.11</artifactId>
      <version>2.1.0</version>
    </dependency>

  然后便是照抄官方文档里的hello raster示例。

package demo

import geotrellis.raster._
import geotrellis.raster.mapalgebra.focal.Square
import geotrellis.spark._

object Main {
  def helloSentence = "Hello GeoTrellis"

  def helloRaster(): Unit = {
    val nd = NODATA    //-2147483648

    val input = Array[Int](
      nd, 7, 1, 1, 3, 5, 9, 8, 2,
      9, 1, 1, 2, 2, 2, 4, 3, 5,
      3, 8, 1, 3, 3, 3, 1, 2, 2,
      2, 4, 7, 1, nd, 1, 8, 4, 3)

    //将数组转化为4*9矩阵
    val iat = IntArrayTile(input, 9, 4)

    //用一个n*n的窗口对矩阵做卷积,设中心值为平均值
    //Square(i) => n = 2 * i + 1
    val focalNeighborhood = Square(1)
    println(focalNeighborhood)
    val meanTile = iat.focalMean(focalNeighborhood)

    for (i <- 0 to 3) {
      for (j <- 0 to 8) {
        print(meanTile.getDouble(j, i) + " ")
      }
      println()
    }
  }

  def main(args: Array[String]): Unit = {
    helloRaster()
  }
}

  输出结果如下

 0  0  0 
 0  0  0 
 0  0  0 

5.666666666666667 3.8 2.1666666666666665 1.6666666666666667 2.5 4.166666666666667 5.166666666666667 5.166666666666667 4.5 
5.6 3.875 2.7777777777777777 1.8888888888888888 2.6666666666666665 3.5555555555555554 4.111111111111111 4.0 3.6666666666666665 
4.5 4.0 3.111111111111111 2.5 2.125 3.0 3.111111111111111 3.5555555555555554 3.1666666666666665 
4.25 4.166666666666667 4.0 3.0 2.2 3.2 3.1666666666666665 3.3333333333333335 2.75 

2.读本地Geotiff文件

  一共四行代码

import geotrellis.raster.io.geotiff.reader.GeoTiffReader
import geotrellis.raster.io.geotiff._

val path: String = "filepath/filename.tif"
val geoTiff: SinglebandGeoTiff = GeoTiffReader.readSingleband(path)

  这种方法用于读取但波段tif影像,要读取多波段影像,可以用

val geoTiff: MultibandGeoTiff = GeoTiffReader.readMultiband(path)

  如果强行用GeoTiffReader.readSingleband(path)方法去读取一个多波段影像,则最后的返回结果是一个单波段影像,且其中的数据为原始影像中的第一个波段
  此外,也可以用影像路径为参数直接构造一个Geotiff对象

import geotrellis.raster.io.geotiff.SinglebandGeoTiff

val geoTiff: SinglebandGeoTiff = SinglebandGeoTiff(path)

3.ETL工具

3.1 调用工具,生成金字塔图层

  这个工具貌似是用来进行数据转换的,试着按照各路文档做了一个将landsat8的tif影像打成金字塔的过程。需要在maven中引入

<!-- https://mvnrepository.com/artifact/org.locationtech.geotrellis/geotrellis-spark-etl -->
    <dependency>
      <groupId>org.locationtech.geotrellis</groupId>
      <artifactId>geotrellis-spark-etl_2.11</artifactId>
      <version>2.1.0</version>
    </dependency>

  函数主体很短

def main(args: Array[String]): Unit = {
    var args = Array[String](
      "--input",
      "D:\\Javaworkspace\\geotrellis\\study\\testdata\\input.json",
      "--output",
      "D:\\Javaworkspace\\geotrellis\\study\\testdata\\output.json",
      "--backend-profiles",
      "D:\\Javaworkspace\\geotrellis\\study\\testdata\\backend-profiles.json"
    );
    Logger.getLogger("org").setLevel(Level.ERROR)
    implicit val sc = SparkUtils.createSparkContext("ETL", new SparkConf(true).setMaster("local[*]"))
    try {
      Etl.ingest[ProjectedExtent, SpatialKey, Tile](args)
    } finally {
      sc.stop
    }
  }

  重点是args里面指定的三个json文件的配置,首先是input.json

[{
    "format":"geotiff",
    "name":"etlTest1",
    "cache":"NONE",
    "backend":{
        "type":"hadoop",
        "path":"D:\\Javaworkspace\\geotrellis\\study\\testdata\\hzpan.TIF"
    }
}]

  这里面制定了输入数据的格式(geotiff),要输出的图层名(etlTest1),读取的方式(type)和数据路径(path)等参数。对于output.json

{
    "backend": {
        "type": "file",
        "path": "D:\\Javaworkspace\\geotrellis\\study\\testdata\\output1"
    },
    "reprojectMethod": "buffered",
    "pyramid": true,
    "tileSize": 256,
    "keyIndexMethod": {
        "type": "zorder"
    },
    "resampleMethod": "nearest-neighbor",
    "layoutScheme": "zoomed",
    "crs": "EPSG:4326"
}

  我的windows本地跑spark和hadoop有个问题一直没解决,导致无法用hadoop的输出接口,因此将type属性指定为file。其他属性都可根据实际情况配置。这里官方文档有一个坑,当pyramid设置为true的时候,resampleMethod就不能设置为cubic-spline,否则会报错。最后是backend-profiles.json

{
    "backend-profiles":[]
}

  貌似是当你使用了accumulo或cassandra来进行输入输出的时候需要配置,一般情况下设个空数组就行。运行成功后在输出目录下新建了两个文件夹

  • attributes
  • etlTest1
      第一个文件夹里存放的似乎是切片后每一个层级的元数据信息,第二个文件夹的命名来自于input.json中的name,其中存放了切片后的图层文件。
      然而,上面的代码只能用于处理单波段影响,即便输入的是多波段影像,该工具也只会对第一个波段进行转换。如果想要处理多波段影像,至少应该修改两个地方:
    (1)首先是input.json的format:
"format":"multiband-geotiff"

 (2)在调用ETL的核心代码中,需要修改切片类型(Tile -> MultibandTile):

Etl.ingest[ProjectedExtent, SpatialKey, MultibandTile](args)

3.2 渲染切片

  上一步中生成的切片文件不能用普通查看图片的方法打开,需要借助Geotrellis提供的类,发布成地图服务或者渲染成图片格式文件(如jpg、png等)。这一步中,尝试将其渲染为图片。
  第一步是读取图层。Geotrellis支持多种读取方式,如Accumulo、h3等,这里使用最基本的文件读取。

    val zoomId = 9  //要读取的图层层级
    implicit val sc = SparkUtils.createSparkContext("ReadLayer", new SparkConf(true).setMaster("local[*]"))
    val path = "D:\\Javaworkspace\\geotrellis\\study\\testdata\\output1"  //图层文件根目录
    val store = FileAttributeStore(path)
    val reader = FileLayerReader(path)
    val layerId = LayerId("etlTest1", zoomId)  //设置图层名称
    val layers: TileLayerRDD[SpatialKey] = reader.read[SpatialKey, Tile, TileLayerMetadata[SpatialKey]](layerId)

  正确读进来,layers将会是个TileLayerRDD对象。\color{red}{重点},这里有个坑,在import的时候,要把geotrellis.spark.io下的所有类文件都引进来,不要只引用到的AttributeStore, LayerReader,否则会报找不到隐式参数的错。

//错误示范,会提示 no implicits found for parameter
import geotrellis.spark.io.{AttributeStore, LayerReader}
//正确方法
import geotrellis.spark.io._

  获取到RDD后,可以利用spark的foreach算子将其输出到本地的文件目录中

    //定义色带,非必须
    val colorMap1 = ColorMap(Map(
          0 -> RGB(0,0,0),
          1 -> RGB(255,255,255)
      ))
    val colorRamp = ColorRamp(RGB(0,0,0), RGB(255,255,255))
        .stops(100)
        .setAlphaGradient(0xFF, 0xAA)
    val outputPath = "D:\\Javaworkspace\\geotrellis\\study\\testdata\\output1\\render\\" + zoomId  //图片输出路径
    val zoomDir: File = new File(outputPath)
    if (!zoomDir.exists()) {
      zoomDir.mkdirs()
    }
    layers.foreach(layer => {
      val key = layer._1
      val tile = layer._2
      val layerPath = outputPath + "\\" + key.row + "_" + key.col + ".jpg"
      tile.renderJpg(colorRamp).write(layerPath)  //调用渲染方法,colorRamp为非必须参数
    })
    sc.stop()

  程序正确运行后,在相应的输出目录可以看到如下文件


图层渲染结果

  将上面的代码稍加修改,就可以将切片导出成tiff影像。首先,需要构造一个GeoTiff对象,它有这样一个构造函数

def apply(tile: Tile, extent: Extent, crs: CRS): SinglebandGeoTiff =
    SinglebandGeoTiff(tile, extent, crs)

  tile类型的对象可以像上面一样通过RDD的foreach算子取得,CRS参数表示坐标系,可以通过EPSG取得,也可以自己构造,同时在geotrellis里也有预定义几个:

/**
 * 4326
 */
object LatLng extends CRS with ObjectNameToString {
  lazy val proj4jCrs = factory.createFromName("EPSG:4326")

  def epsgCode: Option[Int] = Some(4326)
}
/**
 * 3857
 */
object WebMercator extends CRS with ObjectNameToString {
  lazy val proj4jCrs = factory.createFromName("EPSG:3857")

  def epsgCode: Option[Int] = Some(3857)
}
/**
 * Sinusoidal projection, commonly used with MODIS-based data products.
 */
object Sinusoidal extends CRS with ObjectNameToString {
  lazy val proj4jCrs: CoordinateReferenceSystem = factory.createFromParameters(null,
    "+proj=sinu +lon_0=0 +x_0=0 +y_0=0 +a=6371007.181 +b=6371007.181 +units=m +no_defs")
  val epsgCode: Option[Int] = None
}

  最后的Extent对象,也可以通过TileLayerRDD取得,但首先需要取得该切片对应的图层定义:

val layoutDefinition: LayoutDefinition = layers.metadata.layout

  layoutDefinition中有一个mapTransform方法,传入一个SpatialKey,就能得到其对应的格网范围Extend,因此,将字节码图层文件转换为带有空间信息的tiff影像的代码如下:

import geotrellis.raster.io.geotiff.GeoTiff

val layers: TileLayerRDD[SpatialKey] = /** 读取图层代码同上,此处略去 **/
val layoutDefinition: LayoutDefinition = layers.metadata.layout

layers.foreach(layer => {
  val key = layer._1
  val tile = layer._2
  val tiffPath = outputPath + "\\" + key.row + "_" + key.col + ".tiff
  GeoTiff(tile, layoutDefinition.mapTransform(key), LatLng).write(tiffPath)
})

  采用的是4326即WGS84坐标,输出结果与上面的渲染成jpg类似,但具有空间信息,能被EnVI等遥感影像处理软件正确加载。

4. GeoJson的序列化与反序列化

    import geotrellis.vector.{Polygon}
    import geotrellis.vector.io._
    import geotrellis.vector.io.json._
    //serializing
    val json = Polygon((10.0,10.0),(10.0,20.0),(30.0,30.0),(10.0,10.0)).toGeoJson
    println(json)
    //deserializing
    val fc: String = """{
                       |  "type": "FeatureCollection",
                       |  "features": [
                       |    {
                       |      "type": "Feature",
                       |      "geometry": { "type": "Point", "coordinates": [1.0, 2.0] },
                       |      "properties": { "someProp": 14 },
                       |      "id": "target_12a53e"
                       |    }, {
                       |      "type": "Feature",
                       |      "geometry": { "type": "Point", "coordinates": [2.0, 7.0] },
                       |      "properties": { "someProp": 5 },
                       |      "id": "target_32a63e"
                       |    }
                       |  ]
                       |}""".stripMargin
    val geos = fc.parseGeoJson[JsonFeatureCollectionMap]
    val points1 = geos.getAllPoints()
    for (k <- points1.keySet){
      println(k + ": " + points1.get(k).get)
    }

  输出结果如下

{"type":"Polygon","coordinates":[[[10.0,10.0],[10.0,20.0],[30.0,30.0],[10.0,10.0]]]}
target_12a53e: POINT (1 2)
target_32a63e: POINT (2 7)

未完待续

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容