如何在 python 中 write/create GeoTIFF RGB 图像文件?
How do I write/create a GeoTIFF RGB image file in python?
我有 5 个形状为 nx、ny 的 numpy 数组
lons.shape = (nx,ny)
lats.shape = (nx,ny)
reds.shape = (nx,ny)
greens.shape = (nx,ny)
blues.shape = (nx,ny)
红色、绿色和蓝色数组包含的值介于 0–255 之间,lat/lon 数组是 latitude/longitude 像素坐标。
我的问题是如何将这些数据写入 geotiff?
我最终想使用底图绘制图像。
这是我到目前为止的代码,但是我得到了一个巨大的 GeoTIFF 文件(~500MB)并且它是空白的(只是一个黑色图像)。还要注意 nx, ny = 8120, 5416.
from osgeo import gdal
from osgeo import osr
import numpy as np
import h5py
import os
os.environ['GDAL_DATA'] = "/Users/andyprata/Library/Enthought/Canopy_64bit/User/share/gdal"
# read in data
input_path = '/Users/andyprata/Desktop/modisRGB/'
with h5py.File(input_path+'red.h5', "r") as f:
red = f['red'].value
lon = f['lons'].value
lat = f['lats'].value
with h5py.File(input_path+'green.h5', "r") as f:
green = f['green'].value
with h5py.File(input_path+'blue.h5', "r") as f:
blue = f['blue'].value
# convert rgbs to uint8
r = red.astype('uint8')
g = green.astype('uint8')
b = blue.astype('uint8')
# set geotransform
nx = red.shape[0]
ny = red.shape[1]
xmin, ymin, xmax, ymax = [lon.min(), lat.min(), lon.max(), lat.max()]
xres = (xmax - xmin) / float(nx)
yres = (ymax - ymin) / float(ny)
geotransform = (xmin, xres, 0, ymax, 0, -yres)
# create the 3-band raster file
dst_ds = gdal.GetDriverByName('GTiff').Create('myGeoTIFF.tif', ny, nx, 3, gdal.GDT_Float32)
dst_ds.SetGeoTransform(geotransform) # specify coords
srs = osr.SpatialReference() # establish encoding
srs.ImportFromEPSG(3857) # WGS84 lat/long
dst_ds.SetProjection(srs.ExportToWkt()) # export coords to file
dst_ds.GetRasterBand(1).WriteArray(r) # write r-band to the raster
dst_ds.GetRasterBand(2).WriteArray(g) # write g-band to the raster
dst_ds.GetRasterBand(3).WriteArray(b) # write b-band to the raster
dst_ds.FlushCache() # write to disk
dst_ds = None # save, close
我认为问题在于您在创建数据集时传递了它 GDT_Float32。对于像素范围为 0-255 的标准图像,您需要 GDT_Byte。 Float 通常要求值在 0-1 之间。
我拿了你的代码并随机生成了一些数据,这样我就可以测试你的其余部分 API。然后我在太浩湖周围创建了一些虚拟坐标。
这是代码。
#!/usr/bin/env python
from osgeo import gdal
from osgeo import osr
import numpy as np
import os, sys
# Initialize the Image Size
image_size = (400,400)
# Choose some Geographic Transform (Around Lake Tahoe)
lat = [39,38.5]
lon = [-120,-119.5]
# Create Each Channel
r_pixels = np.zeros((image_size), dtype=np.uint8)
g_pixels = np.zeros((image_size), dtype=np.uint8)
b_pixels = np.zeros((image_size), dtype=np.uint8)
# Set the Pixel Data (Create some boxes)
for x in range(0,image_size[0]):
for y in range(0,image_size[1]):
if x < image_size[0]/2 and y < image_size[1]/2:
r_pixels[y,x] = 255
elif x >= image_size[0]/2 and y < image_size[1]/2:
g_pixels[y,x] = 255
elif x < image_size[0]/2 and y >= image_size[1]/2:
b_pixels[y,x] = 255
else:
r_pixels[y,x] = 255
g_pixels[y,x] = 255
b_pixels[y,x] = 255
# set geotransform
nx = image_size[0]
ny = image_size[1]
xmin, ymin, xmax, ymax = [min(lon), min(lat), max(lon), max(lat)]
xres = (xmax - xmin) / float(nx)
yres = (ymax - ymin) / float(ny)
geotransform = (xmin, xres, 0, ymax, 0, -yres)
# create the 3-band raster file
dst_ds = gdal.GetDriverByName('GTiff').Create('myGeoTIFF.tif', ny, nx, 3, gdal.GDT_Byte)
dst_ds.SetGeoTransform(geotransform) # specify coords
srs = osr.SpatialReference() # establish encoding
srs.ImportFromEPSG(3857) # WGS84 lat/long
dst_ds.SetProjection(srs.ExportToWkt()) # export coords to file
dst_ds.GetRasterBand(1).WriteArray(r_pixels) # write r-band to the raster
dst_ds.GetRasterBand(2).WriteArray(g_pixels) # write g-band to the raster
dst_ds.GetRasterBand(3).WriteArray(b_pixels) # write b-band to the raster
dst_ds.FlushCache() # write to disk
dst_ds = None
这是输出。 (注意:目标是产生颜色,不是地形!)
这是 QGIS 中的图像,正在验证投影。
我有 5 个形状为 nx、ny 的 numpy 数组
lons.shape = (nx,ny)
lats.shape = (nx,ny)
reds.shape = (nx,ny)
greens.shape = (nx,ny)
blues.shape = (nx,ny)
红色、绿色和蓝色数组包含的值介于 0–255 之间,lat/lon 数组是 latitude/longitude 像素坐标。
我的问题是如何将这些数据写入 geotiff?
我最终想使用底图绘制图像。
这是我到目前为止的代码,但是我得到了一个巨大的 GeoTIFF 文件(~500MB)并且它是空白的(只是一个黑色图像)。还要注意 nx, ny = 8120, 5416.
from osgeo import gdal
from osgeo import osr
import numpy as np
import h5py
import os
os.environ['GDAL_DATA'] = "/Users/andyprata/Library/Enthought/Canopy_64bit/User/share/gdal"
# read in data
input_path = '/Users/andyprata/Desktop/modisRGB/'
with h5py.File(input_path+'red.h5', "r") as f:
red = f['red'].value
lon = f['lons'].value
lat = f['lats'].value
with h5py.File(input_path+'green.h5', "r") as f:
green = f['green'].value
with h5py.File(input_path+'blue.h5', "r") as f:
blue = f['blue'].value
# convert rgbs to uint8
r = red.astype('uint8')
g = green.astype('uint8')
b = blue.astype('uint8')
# set geotransform
nx = red.shape[0]
ny = red.shape[1]
xmin, ymin, xmax, ymax = [lon.min(), lat.min(), lon.max(), lat.max()]
xres = (xmax - xmin) / float(nx)
yres = (ymax - ymin) / float(ny)
geotransform = (xmin, xres, 0, ymax, 0, -yres)
# create the 3-band raster file
dst_ds = gdal.GetDriverByName('GTiff').Create('myGeoTIFF.tif', ny, nx, 3, gdal.GDT_Float32)
dst_ds.SetGeoTransform(geotransform) # specify coords
srs = osr.SpatialReference() # establish encoding
srs.ImportFromEPSG(3857) # WGS84 lat/long
dst_ds.SetProjection(srs.ExportToWkt()) # export coords to file
dst_ds.GetRasterBand(1).WriteArray(r) # write r-band to the raster
dst_ds.GetRasterBand(2).WriteArray(g) # write g-band to the raster
dst_ds.GetRasterBand(3).WriteArray(b) # write b-band to the raster
dst_ds.FlushCache() # write to disk
dst_ds = None # save, close
我认为问题在于您在创建数据集时传递了它 GDT_Float32。对于像素范围为 0-255 的标准图像,您需要 GDT_Byte。 Float 通常要求值在 0-1 之间。
我拿了你的代码并随机生成了一些数据,这样我就可以测试你的其余部分 API。然后我在太浩湖周围创建了一些虚拟坐标。
这是代码。
#!/usr/bin/env python
from osgeo import gdal
from osgeo import osr
import numpy as np
import os, sys
# Initialize the Image Size
image_size = (400,400)
# Choose some Geographic Transform (Around Lake Tahoe)
lat = [39,38.5]
lon = [-120,-119.5]
# Create Each Channel
r_pixels = np.zeros((image_size), dtype=np.uint8)
g_pixels = np.zeros((image_size), dtype=np.uint8)
b_pixels = np.zeros((image_size), dtype=np.uint8)
# Set the Pixel Data (Create some boxes)
for x in range(0,image_size[0]):
for y in range(0,image_size[1]):
if x < image_size[0]/2 and y < image_size[1]/2:
r_pixels[y,x] = 255
elif x >= image_size[0]/2 and y < image_size[1]/2:
g_pixels[y,x] = 255
elif x < image_size[0]/2 and y >= image_size[1]/2:
b_pixels[y,x] = 255
else:
r_pixels[y,x] = 255
g_pixels[y,x] = 255
b_pixels[y,x] = 255
# set geotransform
nx = image_size[0]
ny = image_size[1]
xmin, ymin, xmax, ymax = [min(lon), min(lat), max(lon), max(lat)]
xres = (xmax - xmin) / float(nx)
yres = (ymax - ymin) / float(ny)
geotransform = (xmin, xres, 0, ymax, 0, -yres)
# create the 3-band raster file
dst_ds = gdal.GetDriverByName('GTiff').Create('myGeoTIFF.tif', ny, nx, 3, gdal.GDT_Byte)
dst_ds.SetGeoTransform(geotransform) # specify coords
srs = osr.SpatialReference() # establish encoding
srs.ImportFromEPSG(3857) # WGS84 lat/long
dst_ds.SetProjection(srs.ExportToWkt()) # export coords to file
dst_ds.GetRasterBand(1).WriteArray(r_pixels) # write r-band to the raster
dst_ds.GetRasterBand(2).WriteArray(g_pixels) # write g-band to the raster
dst_ds.GetRasterBand(3).WriteArray(b_pixels) # write b-band to the raster
dst_ds.FlushCache() # write to disk
dst_ds = None
这是输出。 (注意:目标是产生颜色,不是地形!)
这是 QGIS 中的图像,正在验证投影。