首页 > 编程语言 >Python GDAL/OGR常用样例代码

Python GDAL/OGR常用样例代码

时间:2022-11-21 13:35:17浏览次数:43  
标签:layer target Python 样例 OGR file input output data

目录

安装

推荐使用conda安装python gdal环境,先查询gdal可用版本,再指定版本号,按需安装对应的gdal。

conda search gdal
conda install gdal=version_number

案例

import相应模块和常量定义

from osgeo import gdal, ogr, osr, gdalconst

DRIVER_SHAPE = "ESRI Shapefile"
DRIVER_GTIFF = "GTiff"
EPSG_WGS84 = 4326

矢量转栅格

def vector2raster(input_file, ouput_file, template_file, field="", all_touch="False"):
    """
    矢量图层转栅格

    Params:
        input_file: 输入矢量文件
        output_file: 输出栅格文件
        template_file: 模板栅格文件
        field: 输出栅格所用的input_file字段
        all_touch: False仅矢量中心所在的栅格设置像元值,True只要和矢量相交的栅格都设置像元值

    Ref: https://gdal.org/api/gdal_alg.html#_CPPv419GDALRasterizeLayers12GDALDatasetHiPiiP9OGRLayerH19GDALTransformerFuncPvPdPPc16GDALProgressFuncPv
    """
    # 打开模板栅格
    data = gdal.Open(template_file, gdalconst.GA_ReadOnly)
    # 确定栅格大小
    x_size = data.RasterXSize
    y_size = data.RasterYSize
    # 打开矢量vec_layer
    vector = ogr.GetDriverByName(DRIVER_SHAPE).Open(input_file)
    vec_layer = vector.GetLayer()
    feat_count = vec_layer.GetFeatureCount()
    # 创建输出的tiff栅格文件
    target = gdal.GetDriverByName(DRIVER_GTIFF).Create(ouput_file, x_size, y_size, 1, gdal.GDT_Byte)
    # 设置栅格坐标系与投影
    target.SetGeoTransform(data.GetGeoTransform())
    target.SetProjection(data.GetProjection())

    if field:
        gdal.RasterizeLayer(target, [1], vec_layer, 
        options=["ALL_TOUCHED="+all_touch, "ATTRIBUTE="+field])
    else:
        burn_values = [1] # 如果没有指定属性字段,则使用burn_value作为输出的像素值
        gdal.RasterizeLayer(target, [1], vec_layer, burn_values=burn_values, options=["ALL_TOUCHED="+all_touch])

    # 设置NoData
    NoData_value = 0
    target.GetRasterBand(1).SetNoDataValue(NoData_value)
    target.GetRasterBand(1).FlushCache()
    target = None


# 调用
vector2raster(
    input_file="data/input/vec.shp", 
    ouput_file="data/output/vec2raster.tif", 
    template_file="data/input/template.tif", 
    field="Height"
)

栅格转矢量(多边形)

def raster2polygon(input_raster, output_file, layer_name):
    """
    栅格转矢量多边形

    Params:
        input_raster: 输入栅格文件
        output_file: 输出shapefile文件
        layer_name: 矢量图层名称

    Ref: https://gdal.org/api/gdal_alg.html#_CPPv415GDALFPolygonize15GDALRasterBandH15GDALRasterBandH9OGRLayerHiPPc16GDALProgressFuncPv
    """
    data = gdal.Open(input_raster)
    src_band = data.GetRasterBand(1)
    srs = osr.SpatialReference()
    srs.ImportFromWkt(data.GetProjection()) # 矢量的空间参考和栅格保持一致
    target = ogr.GetDriverByName(DRIVER_SHAPE).CreateDataSource(output_file)
    target_layer = target.CreateLayer(layer_name, srs=srs, geom_type=ogr.wkbPolygon)
    # 给目标shp文件添加一个字段,存储原始栅格的pixel value
    field = ogr.FieldDefn('value',ogr.OFTReal)  
    target_layer.CreateField(field)
    gdal.Polygonize(src_band, src_band, target_layer, 0, [])
    target = None


# 调用
raster2polygon(
    input_raster="data/input/raster.tif",
    output_file="data/output/poly.shp",
    layer_name="poly",
)

矢量叠加

def intersect(input_file1, input_file2, output_file, output_layer_name=""):
    """
    矢量叠加

    Params:
        input_file1: 输入矢量文件1
        input_file2: 输入矢量文件2
        output_file: 输出矢量文件路径
        output_layer_name: 输出图层名
    """
    driver = ogr.GetDriverByName(DRIVER_SHAPE)
    shp1 = driver.Open(input_file1, gdalconst.GA_ReadOnly)
    shp2 = driver.Open(input_file2, gdalconst.GA_ReadOnly)
    src_layer1 = shp1.GetLayer()
    src_layer2 = shp2.GetLayer()
    srs1 = src_layer1.GetSpatialRef()
    srs2 = src_layer2.GetSpatialRef()
    if srs1.GetAttrValue('AUTHORITY',1) != srs2.GetAttrValue('AUTHORITY',1):
        print("空间参考不一致!")
        return
    
    target_ds = ogr.GetDriverByName(DRIVER_SHAPE).CreateDataSource(output_file)
    target_layer = target_ds.CreateLayer(output_layer_name, srs1, geom_type=ogr.wkbPolygon, options=["ENCODING=UTF-8"]) # 设置编码为UFT-8,防止中文出现乱码

    for feat1 in src_layer1:
        geom1 = feat1.GetGeometryRef()
        for feat2 in src_layer2:
            geom2 = feat2.GetGeometryRef()
            if not geom1.Intersects(geom2):
                continue
            intersect = geom1.Intersection(geom2)
            feature = ogr.Feature(target_layer.GetLayerDefn())
            feature.SetGeometry(intersect)
            target_layer.CreateFeature(feature)

    # 清理引用
    target_layer = None
    ds = None


# 调用
intersect(
    input_file1="data/output/vec1.shp",
    input_file2="data/output/vec2.shp",
    output_file="data/output/intersect.shp",
    output_layer_name="intersect",
)

矢量擦除

def erase(erased_file, eraser_file, output_file, output_layer_name=""):
    """
    矢量擦除

    Params:
        erased_file: 被擦除的矢量文件路径
        eraser_file: 擦除矢量文件路径
        output_file: 输出文件路径
        output_layer_name: 输出图层名
    """
    driver = ogr.GetDriverByName(DRIVER_SHAPE)
    shp1 = driver.Open(erased_file, gdalconst.GA_ReadOnly)
    shp2 = driver.Open(eraser_file, gdalconst.GA_ReadOnly)
    src_layer1 = shp1.GetLayer()
    src_layer2 = shp2.GetLayer()
    srs1 = src_layer1.GetSpatialRef()
    srs2 = src_layer2.GetSpatialRef()
    if srs1.GetAttrValue('AUTHORITY',1) != srs2.GetAttrValue('AUTHORITY',1):
        print("空间参考不一致!")
        return
    target_ds = ogr.GetDriverByName(DRIVER_SHAPE).CreateDataSource(output_file)
    target_layer = target_ds.CreateLayer(output_layer_name, srs1, geom_type=ogr.wkbPolygon, options=["ENCODING=UTF-8"])
    ds = src_layer1.Erase(src_layer2, target_layer)
    ds = None


# 调用
erase(
    erased_file="data/output/vec1.shp",
    eraser_file="data/output/vec2.shp", 
    output_file="data/output/erase.shp",
)

缓冲区分析(以点为例)

def point_buffer(point, radius, output_file, layer_name):
    """
    对点集建立缓冲区

    Params:
        point: 输入点坐标
        range: 缓冲区半径
        output_file: 缓冲区输出文件路径
        layer_name: 输出图层名
    """
    # 创建图层
    srs = osr.SpatialReference()
    srs.ImportFromEPSG(EPSG_WGS84) # 选择坐标系,注意地理坐标系和投影坐标系的radius单位不同,前者为度,后者为米
    ds = ogr.GetDriverByName(DRIVER_SHAPE).CreateDataSource(output_file)
    target_layer = ds.CreateLayer(layer_name, srs, geom_type=ogr.wkbPolygon, options=["ENCODING=UTF-8"])
    # 创建geometry
    wkt = "POINT (%f %f)" % (point[0], point[1])
    geom = ogr.CreateGeometryFromWkt(wkt)
    poly = geom.Buffer(radius)
    # 创建feature
    feature = ogr.Feature(target_layer.GetLayerDefn())
    feature.SetGeometry(poly)
    target_layer.CreateFeature(feature)

    target_layer = None
    ds = None


# 调用
point_buffer(
    point=(121.531921, 25.013540),
    range=0.1,
    output_file="data/output/buffer.shp",
    layer_name="buffer",
)

视域分析

def generate_viewshed(input_raster, output_file, location):
    """
    视域分析

    Params:
        input_raster: 输入栅格文件
        output_file: 视域分析输出文件
        location: 观察者所在坐标

    Ref: https://gdal.org/api/gdal_alg.html#_CPPv420GDALViewshedGenerate15GDALRasterBandHPKcPKc12CSLConstListddddddddd16GDALViewshedModed16GDALProgressFuncPv22GDALViewshedOutputType12CSLConstList
    """
    raster = gdal.Open(input_raster)
    band = raster.GetRasterBand(1)
    gdal.ViewshedGenerate(
        srcBand=band, 
        driverName=DRIVER_GTIFF, 
        targetRasterName=output_file,
        creationOptions=[], 
        observerX=location[0], 
        observerY=location[1], 
        observerHeight=2, 
        targetHeight=0, 
        visibleVal=255,
        invisibleVal=0, 
        outOfRangeVal=0, 
        noDataVal=0, 
        dfCurvCoeff=0.85714, 
        mode=2,
        maxDistance=0,
    )


# 调用
generate_viewshed(
    input_raster="data/input/raster.tif",
    output_file="data/output/viewshed.tif",
    location=(120.568740, 32.013540),
)

标签:layer,target,Python,样例,OGR,file,input,output,data
From: https://www.cnblogs.com/greyxy/p/16911141.html

相关文章

  • python的协程
    python协程库gevent学习--gevent数据结构及实战(三)gevent学习系列第三章,前面两章分析了大量常用几个函数的源码以及实现原理。这一章重点偏向实战了,按照官方给出的ge......
  • Python爬取酷狗音乐Top500首歌曲并下载到本地
    #@Author:林云#@Time:2022/11/2018:05#@File:KuGouYinyue.py#@Project:PycharmProjectsimportjsonimportosfromtimeimportsleepimportrequestsfromlx......
  • python安装virtualenvwrapper报错解决办法
    在执行sudopipinstall virtualenvwrapper时候,会有一个警告,一个报错1、首先报黄色警告:Thedirectory'/Users/lvxiujun/Library/Caches/pip/http'oritsparentdirect......
  • Python字符串的encode与decode研究心得乱码问题解决方法(转)
    ​​Python字符串的encode与decode研究心得乱码问题解决方法(转)​​为什么会报错“UnicodeEncodeError:'ascii'codeccan'tencodecharactersinposition0-1:o......
  • python的base64
    ​​python3.4.1下base64编码问题​​作者:廖师兄 时间:2014-09-05 分类:​​python​​初学py,学的是3.x版本,今天遇到base64编码问题importbase64encodestr=base......
  • Python psutil模块 Process 类
    psutil模块Process类如果需要了解其他跟多理解的=》Python知识点合集 如果需要查看官方手册解读的=》Process类......
  • 【转载】python的魔法方法———A Guide to Python's Magic Methods
    原文地址:https://rszalski.github.io/magicmethods/     =========================================================== AGuidetoPython'sMagicMetho......
  • python list dict util (分割,分组)
     1.list数据分割为多个小列表 (java  lists.partition)2.分组     importitertoolsdefpartition(mylist,size):""":parammylist:需要分......
  • 使用UDP协议实现简单的分布式日志服务, java和python
    使用UDP协议实现简单的分布式日志服务,java和python这几天系统出现问题,需要查原因.日志分散在各个服务器上,查起来很要命.网上百度了好久,最后发现,各种日志的处理......
  • python代码规范工具
    文章目录​​一:Pycharm自动创建文件头部​​​​二:代码门禁​​​​三:CommitAngular规范​​一:Pycharm自动创建文件头部Pycham—>Preferences—>编辑器—>文件和代......