GDAL 矢量裁剪柵格


  本節將介紹如何在Python中用GDAL實現根據矢量邊界裁剪柵格數據。

from osgeo import gdal, gdal_array
import shapefile
import numpy as np
import os

#批量shp裁剪tiff影像
try:
    import Image
    import ImageDraw
except:
    from PIL import Image, ImageDraw

def read_tiff(inpath):
  ds=gdal.Open(inpath)
  row=ds.RasterXSize
  col=ds.RasterYSize
  band=ds.RasterCount

  data=np.zeros([row,col,band])
  for i in range(band):
   dt=ds.GetRasterBand(1)
   data[:,:,i]=dt.ReadAsArray(0,0,col,row)
  return data

def image2Array(i):
    """
    將一個Python圖像庫的數組轉換為一個gdal_array圖片
    """
    a = gdal_array.numpy.frombuffer(i.tobytes(), 'b')
    a.shape = i.im.size[1], i.im.size[0]
    return a

def world2Pixel(geoMatrix, x, y):
    """
    使用GDAL庫的geomatrix對象((gdal.GetGeoTransform()))計算地理坐標的像素位置
    """
    ulx = geoMatrix[0]
    uly = geoMatrix[3]
    xDist = geoMatrix[1]
    yDist = geoMatrix[5]
    rtnX = geoMatrix[2]
    rtnY = geoMatrix[4]
    pixel = int((x - ulx) / xDist)
    line = int((uly - y) / abs(yDist))
    return (pixel, line)


def write_img(filename,im_proj,im_geotrans,im_data):
    if 'int8' in im_data.dtype.name:
        datatype = gdal.GDT_Byte
    elif 'int16' in im_data.dtype.name:
        datatype = gdal.GDT_UInt16
    else:
        datatype = gdal.GDT_Float32

    if len(im_data.shape) == 3:
        im_bands, im_height, im_width = im_data.shape
    else:
        im_bands, (im_height, im_width) = 1,im_data.shape

    driver = gdal.GetDriverByName("GTiff")
    dataset = driver.Create(filename, im_width, im_height, im_bands, datatype)

    dataset.SetGeoTransform(im_geotrans)
    dataset.SetProjection(im_proj)
    if im_bands == 1:

        dataset.GetRasterBand(1).WriteArray(im_data)
    else:
        for i in range(im_bands):
            dataset.GetRasterBand(i+1).WriteArray(im_data[i])

    del dataset

def sha_raster(raster,shp,output):
    srcArray = gdal_array.LoadFile(raster)
    # 同時載入gdal庫的圖片從而獲取geotransform
    srcImage = gdal.Open(raster)
    geoProj = srcImage.GetProjection()
    geoTrans = srcImage.GetGeoTransform()
    r = shapefile.Reader(shp)
    # 將圖層擴展轉換為圖片像素坐標
    minX, minY, maxX, maxY = r.bbox
    ulX, ulY = world2Pixel(geoTrans, minX, maxY)
    lrX, lrY = world2Pixel(geoTrans, maxX, minY)
    pxWidth = int(lrX - ulX)
    pxHeight = int(lrY - ulY)
    clip = srcArray[:, ulY:lrY, ulX:lrX]
    # 為圖片創建一個新的geomatrix對象以便附加地理參照數據
    geoTrans = list(geoTrans)
    geoTrans[0] = minX
    geoTrans[3] = maxY
    # 在一個空白的8字節黑白掩膜圖片上把點映射為像元繪制市縣
    # 邊界線
    pixels = []
    for p in r.shape(0).points:
        pixels.append(world2Pixel(geoTrans, p[0], p[1]))
    rasterPoly = Image.new("L", (pxWidth, pxHeight), 1)
    # 使用PIL創建一個空白圖片用於繪制多邊形
    rasterize = ImageDraw.Draw(rasterPoly)
    rasterize.polygon(pixels, 0)
    # 使用PIL圖片轉換為Numpy掩膜數組
    mask = image2Array(rasterPoly)
    name = os.path.basename(raster).split(".tif")[0]
    outfile = output + "\\" + name+ "_cut.tif"  # 對輸出文件命名
    # 根據掩膜圖層對圖像進行裁剪
    clip = gdal_array.numpy.choose(mask, (clip, 0)).astype(gdal_array.numpy.uint16)
    write_img(outfile, geoProj, geoTrans, clip)
    gdal.ErrorReset()

if __name__ == "__main__":
    raster = r'D:\test\裁剪實驗\image\15.tif'
    # 用於裁剪的多邊形shp文件
    shp = r'D:\test\裁剪實驗\shp\2.shp'
    # 裁剪后的柵格數據
    output = r'D:\test\裁剪實驗\out'

    #依據shp創建掩膜進行對tiff文件的裁剪
    sha_raster(raster,shp,output)

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM