Convertendo multipolígono enorme em polígonos

8

Eu tenho um shapefile com alguns multipolígonos enormes, com 100.000 partes. Qual seria a maneira mais fácil de dividi-los em polígonos de parte única? Estou procurando algo como a função "Multipart to singlepart" do QGIS, mas o arquivo é grande demais para o QGIS lidar. Suponho que provavelmente já exista algum módulo Python que possa fazer isso por mim. Alguma dica?

leo
fonte
Você tem a opção de carregar a camada no PostGIS?
Get Spatial
Em que sua camada está armazenada agora? Pode ser que a alteração do formato de armazenamento permita que ele funcione no QGIS, apenas considerando as diferenças de eficiência.
Get Spatial
1
Está em um shapefile, então a solução do @ Barrett abaixo é perfeita!
Leo

Respostas:

11

Os arquivos de forma não têm o tipo MultiPolygon (type = Polygon), mas os suportam de qualquer maneira (todos os anéis são armazenados em um polígono = lista de polígonos, veja GDAL: ESRI Shapefile )

É mais fácil com Fiona e Shapely :

import fiona
from shapely.geometry import shape, mapping

# open the original MultiPolygon file
with fiona.open('multipolygons.shp') as source:
    # create the new file: the driver and crs are the same
    # for the schema the geometry type is "Polygon" instead
    output_schema = dict(source.schema)  # make an independant copy
    output_schema['geometry'] = "Polygon"

    with fiona.open('output.shp', 'w', 
                    driver=source.driver,
                    crs=source.crs,
                    schema=output_schema) as output:

        # read the input file
        for multi in source:

           # extract each Polygon feature
           for poly in shape(multi['geometry']):

              # write the Polygon feature
              output.write({
                  'properties': multi['properties'],
                  'geometry': mapping(poly)
              })
gene
fonte
11

da lista de discussão GDAL usando python

import os
from osgeo import ogr

def multipoly2poly(in_lyr, out_lyr):
    for in_feat in in_lyr:
        geom = in_feat.GetGeometryRef()
        if geom.GetGeometryName() == 'MULTIPOLYGON':
            for geom_part in geom:
                addPolygon(geom_part.ExportToWkb(), out_lyr)
        else:
            addPolygon(geom.ExportToWkb(), out_lyr)

def addPolygon(simplePolygon, out_lyr):
    featureDefn = out_lyr.GetLayerDefn()
    polygon = ogr.CreateGeometryFromWkb(simplePolygon)
    out_feat = ogr.Feature(featureDefn)
    out_feat.SetGeometry(polygon)
    out_lyr.CreateFeature(out_feat)
    print 'Polygon added.'

from osgeo import gdal
gdal.UseExceptions()
driver = ogr.GetDriverByName('ESRI Shapefile')
in_ds = driver.Open('data/multipoly.shp', 0)
in_lyr = in_ds.GetLayer()
outputshp = 'data/poly.shp'
if os.path.exists(outputshp):
    driver.DeleteDataSource(outputshp)
out_ds = driver.CreateDataSource(outputshp)
out_lyr = out_ds.CreateLayer('poly', geom_type=ogr.wkbPolygon)
multipoly2poly(in_lyr, out_lyr)
Barrett
fonte