本文主要介绍GeoPandas的使用要点。GeoPandas是一个Python开源项目,旨在提供丰富而简单的地理空间数据处理接口。GeoPandas扩展了Pandas的数据类型,并使用matplotlib进行绘图。GeoPandas官方仓库地址为:GeoPandas。GeoPandas的官方文档地址为:GeoPandas-doc。本文主要参考GeoPandas Examples Gallery。GeoPandas的基础使用见Python绘制数据地图1-GeoPandas入门指北。GeoPandas的可视化入门见Python绘制数据地图2-GeoPandas地图可视化。


(资料图)

本文所有代码见:Python-Study-Notes

GeoPandas推荐使用Python3.7版本及以上,运行环境最好是linux系统。GeoPandas安装命令如下:

pip install geopandas

如果上述命令安装出问题,则推荐使用conda安装GeoPandas,命令如下:

conda install geopandas

或:

conda install --channel conda-forge geopandas

除了GeoPandas需要安装,以下第三方库也需要安装:

pip install mapclassifypip install matplotlib_scalebarpip install rtreepip install contextily
# jupyter notebook环境去除warningimport warningswarnings.filterwarnings("ignore")# 查看geopandas版本import geopandas as gpdgpd.__version__
"0.13.2"
目录1 分级统计图Choropleth2 通过DataFrame创建GeoDataFrame3 添加比例尺4 图层操作与几何运算4.1 图层叠加4.2 空间连接4.3 几何操作4.4 汇总4.5 缺失值与空值处理5 背景地图叠加5.1 简单背景地图叠加5.2 定制化背景地图5.3 离线背景地图6 参考1 分级统计图Choropleth

分级统计图Choropleth是一种表示地理区域内数据分布的可视化图表。它将地图划分为不同的区域,并使用颜色或阴影的不同程度来显示该区域的数据值。通常,分级统计图用于显示人口统计、自然资源分布等数据。分级统计图以帮助观察者更容易地理解数据在地理空间上的分布情况和变化趋势,有助于制定决策和规划相关工作。

import geopandas as gpdfrom geopandas import read_file
# pip install mapclassifyimport mapclassifymapclassify.__version__
"2.5.0"
# 读取四川地图数据,数据来自DataV.GeoAtlas,将其投影到EPSG:4573data = gpd.read_file("https://geo.datav.aliyun.com/areas_v3/bound/510000_full.json").to_crs("EPSG:4573")data.head()
adcodenamechildrenNumlevelparentsubFeatureIndexgeometry
0510100成都市20city{"adcode": 510000}0MULTIPOLYGON (((18399902.859 3356187.915, 1840...
1510300自贡市6city{"adcode": 510000}1MULTIPOLYGON (((18419941.941 3231303.167, 1842...
2510400攀枝花市5city{"adcode": 510000}2MULTIPOLYGON (((18183734.470 2889855.327, 1818...
3510500泸州市7city{"adcode": 510000}3MULTIPOLYGON (((18540813.879 3244247.734, 1853...
4510600德阳市6city{"adcode": 510000}4MULTIPOLYGON (((18516163.207 3398495.768, 1851...

简单分级统计

以下代码通过scheme分级统计四川省各地级市所包含区县数。

ax = data.plot(    column="childrenNum",    scheme="QUANTILES", # 设置分层设色标准    edgecolor="lightgrey",     k=7, # 分级数量    cmap="Blues",    legend=True,    # 通过fmt设置位数    legend_kwds={"loc": "center left", "bbox_to_anchor": (1, 0.5),"fmt": "{:.2f}"})# 显示各地级市包含区县数量for index in data.index:    x = data.iloc[index].geometry.centroid.x    y = data.iloc[index].geometry.centroid.y    name = data.iloc[index]["childrenNum"]    ax.text(x, y, name, ha="center", va="center",color="red")
# 查看label,也就是分级区间labels = [t.get_text() for t in ax.get_legend().get_texts()]labels
[" 3.00,  5.00", " 5.00,  6.00", " 6.00,  6.57", " 6.57,  7.43", " 7.43,  9.29", " 9.29, 13.57", "13.57, 20.00"]
# 查看各个分级标准和区间数量,一般是左开右闭res = mapclassify.Quantiles(data["childrenNum"], k=7)res
Quantiles   Interval      Count----------------------[ 3.00,  5.00] |     5( 5.00,  6.00] |     4( 6.00,  6.57] |     0( 6.57,  7.43] |     3( 7.43,  9.29] |     3( 9.29, 13.57] |     3(13.57, 20.00] |     3

间隔展示

ax = data.plot(    column="childrenNum",    scheme="BoxPlot",     edgecolor="k",    cmap="OrRd", # 设置分层设色标准    legend=True,    # 通过interval设置是否展示区间间隔    legend_kwds={"loc": "center left", "bbox_to_anchor": (1, 0.5), "interval": True})# 显示各地级市包含区县数量for index in data.index:    x = data.iloc[index].geometry.centroid.x    y = data.iloc[index].geometry.centroid.y    name = data.iloc[index]["childrenNum"]    ax.text(x, y, name, ha="center", va="center",color="red")

分类展示

以数值分类的方式展示数据,其中区县数量为20的地级市为成都市。

ax = data.plot(    column="childrenNum",    categorical=True, # 以数值分类的方式展示    legend=True,    cmap="tab20",    # 对于分类数据,fmt设置无用    legend_kwds={"loc": "center left", "bbox_to_anchor": (1, 0.5), "fmt": "{:.0f}"},)  # 显示各地级市包含区县数量for index in data.index:    x = data.iloc[index].geometry.centroid.x    y = data.iloc[index].geometry.centroid.y    name = data.iloc[index]["childrenNum"]    ax.text(x, y, name, ha="center", va="center",color="red")

自定义分级

# 自定义分级标准def custom(value):    # 设置ABC三个等级    level = None    if value > 15:       level = "A"    elif value > 8:        level = "B"    else:        level = "C"    return level# 根据自定义函数映射为新的列data["level"] = data["childrenNum"].apply(custom)data.head()
adcodenamechildrenNumlevelparentsubFeatureIndexgeometry
0510100成都市20A{"adcode": 510000}0MULTIPOLYGON (((18399902.859 3356187.915, 1840...
1510300自贡市6C{"adcode": 510000}1MULTIPOLYGON (((18419941.941 3231303.167, 1842...
2510400攀枝花市5C{"adcode": 510000}2MULTIPOLYGON (((18183734.470 2889855.327, 1818...
3510500泸州市7C{"adcode": 510000}3MULTIPOLYGON (((18540813.879 3244247.734, 1853...
4510600德阳市6C{"adcode": 510000}4MULTIPOLYGON (((18516163.207 3398495.768, 1851...
ax = data.plot(    column="level",    categorical=True, # 以数值分类的方式展示    legend=True,    cmap="coolwarm",    # 对于分类数据,fmt设置无用    legend_kwds={"loc": "center left", "bbox_to_anchor": (1, 0.5), "fmt": "{:.0f}"},)  # 显示各地级市包含区县数量for index in data.index:    x = data.iloc[index].geometry.centroid.x    y = data.iloc[index].geometry.centroid.y    name = data.iloc[index]["childrenNum"]    ax.text(x, y, name, ha="center", va="center",color="red")
2 通过DataFrame创建GeoDataFrame

基于经纬度数据

GeoDataFrame有一个geometry列,我们可以通过经纬度数据Latitude和Longitude创建该列。

import pandas as pd# 生成关于南美城市的dataframe数据df = pd.DataFrame(    {        "City": ["Buenos Aires", "Brasilia", "Santiago", "Bogota", "Caracas"],        "Country": ["Argentina", "Brazil", "Chile", "Colombia", "Venezuela"],        "Latitude": [-34.58, -15.78, -33.45, 4.60, 10.48],        "Longitude": [-58.66, -47.91, -70.66, -74.08, -66.86],    })df
CityCountryLatitudeLongitude
0Buenos AiresArgentina-34.58-58.66
1BrasiliaBrazil-15.78-47.91
2SantiagoChile-33.45-70.66
3BogotaColombia4.60-74.08
4CaracasVenezuela10.48-66.86
# 将dataframe转换为GeoDataFrameimport geopandas as gpdgdf = gpd.GeoDataFrame(    df, geometry=gpd.points_from_xy(df.Longitude, df.Latitude), crs="EPSG:4326")gdf
CityCountryLatitudeLongitudegeometry
0Buenos AiresArgentina-34.58-58.66POINT (-58.66000 -34.58000)
1BrasiliaBrazil-15.78-47.91POINT (-47.91000 -15.78000)
2SantiagoChile-33.45-70.66POINT (-70.66000 -33.45000)
3BogotaColombia4.60-74.08POINT (-74.08000 4.60000)
4CaracasVenezuela10.48-66.86POINT (-66.86000 10.48000)
# 在南美地图上展示world = gpd.read_file(gpd.datasets.get_path("naturalearth_lowres"))# 定位到南美ax = world.cx[-90:-55, -25:15].plot(color="white", edgecolor="black")# 在ax区域上绘制地图gdf.plot(ax=ax, color="red")

基于WTK格式数据

WKT (Well-Known Text) 是一种用于描述地理位置的数据格式。WTK格式的数据包含点、线、多边形等地理位置信息。WTK格式的数据可以被许多GIS软件和地理位置分析工具所读取和处理。我们可以将带有WKT数据的DataFrame转换为GeoDataframe。

df = pd.DataFrame(    {        "City": ["Buenos Aires", "Brasilia", "Santiago", "Bogota", "Caracas"],        "Country": ["Argentina", "Brazil", "Chile", "Colombia", "Venezuela"],        "Coordinates": [            "POINT(-58.66 -34.58)",            "POINT(-47.91 -15.78)",            "POINT(-70.66 -33.45)",            "POINT(-74.08 4.60)",            "POINT(-66.86 10.48)",        ],    })df
CityCountryCoordinates
0Buenos AiresArgentinaPOINT(-58.66 -34.58)
1BrasiliaBrazilPOINT(-47.91 -15.78)
2SantiagoChilePOINT(-70.66 -33.45)
3BogotaColombiaPOINT(-74.08 4.60)
4CaracasVenezuelaPOINT(-66.86 10.48)
# 创建新列然后数据转换df["Coordinates"] = gpd.GeoSeries.from_wkt(df["Coordinates"])gdf = gpd.GeoDataFrame(df, geometry="Coordinates", crs="EPSG:4326")print(gdf.head())
City    Country                  Coordinates0  Buenos Aires  Argentina  POINT (-58.66000 -34.58000)1      Brasilia     Brazil  POINT (-47.91000 -15.78000)2      Santiago      Chile  POINT (-70.66000 -33.45000)3        Bogota   Colombia    POINT (-74.08000 4.60000)4       Caracas  Venezuela   POINT (-66.86000 10.48000)
# 在南美地图上展示world = gpd.read_file(gpd.datasets.get_path("naturalearth_lowres"))# 定位到南美ax = world.cx[-90:-55, -25:15].plot(color="white", edgecolor="black")# 在ax区域上绘制地图gdf.plot(ax=ax, color="red")
3 添加比例尺

在地理空间数据分析和可视化过程中,比例尺可以帮助我们了解地图上的距离和大小关系。基于matplotlib进行可视化时,可以利用matplotlib-scalebar库添加比例尺。

简单比例尺

import geopandas as gpd# pip install matplotlib_scalebar安装from matplotlib_scalebar.scalebar import ScaleBar
# nybb为纽约五大地图nybb = gpd.read_file(gpd.datasets.get_path("nybb"))# 北美地区常见坐标系统,坐标以米为单位nybb = nybb.to_crs(32619)  nybb.head()
BoroCodeBoroNameShape_LengShape_Areageometry
05Staten Island330470.0103321.623820e+09MULTIPOLYGON (((72387.313 4502901.349, 72390.3...
14Queens896344.0477633.045213e+09MULTIPOLYGON (((90672.492 4505050.592, 90663.5...
23Brooklyn741080.5231661.937479e+09MULTIPOLYGON (((88021.476 4503764.521, 87967.7...
31Manhattan359299.0964716.364715e+08MULTIPOLYGON (((76488.408 4515823.054, 76399.6...
42Bronx464392.9918241.186925e+09MULTIPOLYGON (((86828.383 4527641.247, 86816.3...

如下所示,创建ScaleBar对象所需的唯一参数是dx。dx表示输入图片每一个像素代表的长度,units为dx的单位。此参数的值取决于坐标参考系的单位。在前面nybb数据集已经使用epsg:32619坐标系统,该坐标系以单位米为单位,如下所示,可以看到nybb.crs输出结果中Axis Info项标识了该参考系以metre米为单位。

nybb.crs
Name: WGS 84 / UTM zone 19NAxis Info [cartesian]:- E[east]: Easting (metre)- N[north]: Northing (metre)Area of Use:- name: Between 72°W and 66°W, northern hemisphere between equator and 84°N, onshore and offshore. Aruba. Bahamas. Brazil. Canada - New Brunswick (NB); Labrador; Nunavut; Nova Scotia (NS); Quebec. Colombia. Dominican Republic. Greenland. Netherlands Antilles. Puerto Rico. Turks and Caicos Islands. United States. Venezuela.- bounds: (-72.0, 0.0, -66.0, 84.0)Coordinate Operation:- name: UTM zone 19N- method: Transverse MercatorDatum: World Geodetic System 1984 ensemble- Ellipsoid: WGS 84- Prime Meridian: Greenwich

在下面代码中添加了比例尺和像素尺寸,该比例尺采用的是线段式表示方式,即在地图上绘制一条线段并注明该地图上该线段所代表的实际距离。

ax = nybb.plot()# 在地图中添加比例尺和像素尺寸scalebar =ScaleBar(dx=1,units="m")ax.add_artist(scalebar)

确定比例尺基准长度

如下所示,以经纬度为单位的epsg:4326坐标系,其单位尺度为度(经纬度)。

# nybb为纽约五大区地图nybb = gpd.read_file(gpd.datasets.get_path("nybb"))nybb = nybb.to_crs(4326) nybb.plot()
nybb.crs
Name: WGS 84Axis Info [ellipsoidal]:- Lat[north]: Geodetic latitude (degree)- Lon[east]: Geodetic longitude (degree)Area of Use:- name: World.- bounds: (-180.0, -90.0, 180.0, 90.0)Datum: World Geodetic System 1984 ensemble- Ellipsoid: WGS 84- Prime Meridian: Greenwich

可以通过计算该地图中相邻两点之间的距离长度来确定比例尺基准长度,要注意的是这两点应该位于待绘制的地图中。

from shapely.geometry.point import Pointpoints = gpd.GeoSeries(    [Point(-73.9, 40.7), Point(-74.9, 40.7)], crs=4326)  # 将两点转换到以米为单位的坐标系points = points.to_crs(32619)  # 计算点之间的距离,距离单位为坐标系的单位distance_meters = points[0].distance(points[1])distance_meters
84698.53985065906
nybb = nybb.to_crs(4326) ax = nybb.plot()ax.add_artist(ScaleBar(distance_meters,"m"))

比例尺自定义

通过更改 ScaleBar 参数能够调整比例尺的显示效果,ScaleBar具体参数如下所示。这些参数的使用可以自行尝试。

scalebar = ScaleBar(    dx, # 像素和长度之间的比例尺。例如,如果一个像素代表1毫米,则dx=0.001。    units="m", # 长度单位    dimension="si-length", # 维度    label=None, # 刻度尺标签    length_fraction=None, # 刻度尺长度占比    height_fraction=None, # 刻度尺高度占比    width_fraction=None, # 刻度尺宽度占比    location=None, # 刻度尺的位置    pad=None, # 刻度尺和边框之间的间距    border_pad=None, # 刻度尺和边框之间的边距    sep=None, # 刻度尺标签和刻度之间的距离    frameon=None, # 是否显示边框    color=None, # 刻度尺和标签的颜色    box_color=None, # 边框的颜色    box_alpha=None, # 边框的透明度    scale_loc=None, # 刻度线的位置    label_loc=None, # 刻度尺标签的位置    font_properties=None, # 标签和刻度线的字体属性    label_formatter=None, # 标签的格式化函数    scale_formatter=None, # 刻度线的格式化函数    fixed_value=None, # 固定的数值    fixed_units=None, # 固定的单位    animated=False, # 是否允许动画    rotation=None, # 刻度尺的旋转角度    bbox_to_anchor=None, # bbox的锚点    bbox_transform=None, # bbox的变换)

此外,也可以更改一像素代表的长度单位,如ScaleBar(2, dimension="si-length", units="km")表示图中1像素代表实际si-length(国际单位制)中的2km。所支持的长度单位参数如下表所示:

dimensionunits
si-lengthkm, m, cm, um
imperial-lengthin, ft, yd, mi
si-length-reciprocal1/m, 1/cm
angledeg

一些比例尺参数调整的示例如下

nybb = gpd.read_file(gpd.datasets.get_path("nybb")).to_crs(32619)ax = nybb.plot()# 改变位置和方向scale1 = ScaleBar(    dx=1,    label="Scale 1",    location="lower left",  # 位置    label_loc="left",    scale_loc="top",  # 注释文字相对于横线方向位置)# 改变颜色scale2 = ScaleBar(    dx=1,    label="Scale 2",    location="center",    color="#b32400",    box_color="yellow",    box_alpha=0.8,  # 透明度)# 改变文字scale3 = ScaleBar(    dx=1,    label="Scale 3",    font_properties={        "size": "large",    },      location="lower right",  # 位置    scale_formatter=lambda value, unit: f"> {value} {unit} <",)# 改变长度scale4 = ScaleBar(    dx=1,    label="Scale 4",    length_fraction=0.5, # 表示刻度线占绘图区域的50%    scale_loc="top",    label_loc="left",    border_pad=1,    pad=0.25,)ax.add_artist(scale1)ax.add_artist(scale2)ax.add_artist(scale3)ax.add_artist(scale4)
4 图层操作与几何运算4.1 图层叠加

在geopandas中,overlay()函数是用于将两个地理图层进行叠加分析的函数。它可以进行求交集、并集、差集和对称差集等操作。overlay()函数的基本语法如下:

geopandas.overlay(layer1, layer2, how)

其中,layer1和layer2是两个geopandas地理图层对象,how是一个字符串,指定要进行的叠加操作。how参数有以下取值:

intersection:交集union:并集difference:差集symmetric_difference:对称差集identity

在下面的示例中展示overlay函数的使用方式。

准备数据

import geopandas as gpdimport pandas as pdfrom shapely.geometry import Polygon, Point
# 画一个圆center = Point(2, 2)  # 圆心坐标radius = 1  # 圆的半径circle = center.buffer(radius)gdf1 = gpd.GeoDataFrame({"geometry": circle, "circle":[0]})gdf1.plot()
gdf1
geometrycircle
0POLYGON ((3.00000 2.00000, 2.99518 1.90198, 2....0
# 画两个正方形square = gpd.GeoSeries([Polygon([(0, 0), (2, 0), (2, 2), (0, 2)]),                        Polygon([(2, 2), (4, 2), (4, 4), (2, 4)])])gdf2 = gpd.GeoDataFrame({"geometry": square, "square":[0,1]})gdf2.plot()
gdf2
geometrysquare
0POLYGON ((0.00000 0.00000, 2.00000 0.00000, 2....0
1POLYGON ((2.00000 2.00000, 4.00000 2.00000, 4....1
# 展示共同绘图结果ax = gdf1.plot()gdf2.plot(ax=ax)

交集intersection

# 需要pip install rtreegdf = gpd.overlay(gdf1, gdf2, how="intersection")gdf
circlesquaregeometry
000POLYGON ((1.90198 1.00482, 1.80491 1.01921, 1....
101POLYGON ((2.09802 2.99518, 2.19509 2.98079, 2....
gdf.plot(cmap="tab10")

并集union

# 需要pip install rtreegdf = gpd.overlay(gdf1, gdf2, how="union")gdf
circlesquaregeometry
00.00.0POLYGON ((1.90198 1.00482, 1.80491 1.01921, 1....
10.01.0POLYGON ((2.09802 2.99518, 2.19509 2.98079, 2....
20.0NaNMULTIPOLYGON (((1.00000 2.00000, 1.00482 2.098...
3NaN0.0POLYGON ((2.00000 0.00000, 0.00000 0.00000, 0....
4NaN1.0POLYGON ((2.00000 4.00000, 4.00000 4.00000, 4....
gdf.plot(cmap="tab10")

差集difference

# 需要pip install rtree# 提取在gdf1中,但不在gdf2中的区域gdf = gpd.overlay(gdf1, gdf2, how="difference")# 也可以用以下写法更加直观# gdf = gdf1.overlay(gdf2, how="difference")gdf
geometrycircle
0MULTIPOLYGON (((1.00000 2.00000, 1.00482 2.098...0
gdf.plot(cmap="tab10")

对称差集symmetric_difference

# 需要pip install rtree# 提取不在gdf1和pdf2交集的区域gdf = gpd.overlay(gdf1, gdf2, how="symmetric_difference")gdf
circlesquaregeometry
00.0NaNMULTIPOLYGON (((1.00000 2.00000, 1.00482 2.098...
1NaN0.0POLYGON ((2.00000 0.00000, 0.00000 0.00000, 0....
2NaN1.0POLYGON ((2.00000 4.00000, 4.00000 4.00000, 4....
gdf.plot(cmap="tab10")

identity

identity是ArcGIS中常用的操作。意思是将源地理图层与参考图层进行比较,以在源图层中标识与参考图层中相交的区域。使用identity的一个典型场景是当需要分析两个图层交集的时候。例如,可能有一个图层包含了所有的道路,另一个图层包含了所有的建筑。通过使用identity可以找到所有的建筑物位于哪些道路上。

# 需要pip install rtreegdf = gpd.overlay(gdf1, gdf2, how="identity")gdf
circlesquaregeometry
00.00.0POLYGON ((1.90198 1.00482, 1.80491 1.01921, 1....
10.01.0POLYGON ((2.09802 2.99518, 2.19509 2.98079, 2....
20.0NaNMULTIPOLYGON (((1.00000 2.00000, 1.00482 2.098...
gdf.plot(cmap="tab10")
4.2 空间连接

空间连接允许将两个或多个空间数据集合并成一个新的数据集。例如,我们有两个数据集,一个包含所有城市的边界,另一个包含所有的人口数据。通过空间连接,我们可以将这两个数据集合并成一个新的数据集,其中每个城市都会有相应的人口数据。GeoPandas提供sjoin函数将两个GeoDataFrame数据集基于空间关系进行连接。sjoin函数常用参数如下:

sjoin(left_df, right_df, how="inner", op="intersects", lsuffix="left", rsuffix="right")

其中,参数含义如下:

left_df:左侧的GeoDataFrame数据集。right_df:右侧的GeoDataFrame数据集。how:连接方式,可选项如下:inner(默认选项):返回两个GeoDataFrame中具有共同空间索引的几何体的交集。left:返回左侧GeoDataFrame中的所有几何体,以及右侧GeoDataFrame中与之相交的几何体。如果右侧GeoDataFrame中没有与左侧相交的几何体,则右侧数据中的所有列都将为null。right:与left相反,返回右侧GeoDataFrame中的所有几何体,以及左侧GeoDataFrame中与之相交的几何体。如果左侧GeoDataFrame中没有与右侧相交的几何体,则左侧数据中的所有列都将为null。predicate:连接的空间关系,常用选项如下:intersects(默认选项):返回两个几何体相交的所有几何体。contains:返回左侧GeoDataFrame中包含于右侧GeoDataFrame中的所有几何体。within:返回右侧GeoDataFrame中包含于左侧GeoDataFrame中的所有几何体。touches:返回两个几何体相切的所有几何体。crosses:返回两个几何体相交但不相切的所有几何体。overlaps:返回两个几何体部分重叠的所有几何体。lsuffix:组合后左侧数据集中几何对象列的后缀,默认为leftrsuffix:组合后右侧数据集中几何对象列的后缀,默认为right

以下示例展示了如何使用sjoin函数进行空间连接。

准备数据

# 创建点 GeoDataFramepoints = gpd.GeoDataFrame(    [        {"id": "p1", "geometry": Point(0, 0)},        {"id": "p2", "geometry": Point(1, 1)},        {"id": "p3", "geometry": Point(2, 2)},        {"id": "p4", "geometry": Point(3, 3)}    ],    crs="EPSG:4326")points
idgeometry
0p1POINT (0.00000 0.00000)
1p2POINT (1.00000 1.00000)
2p3POINT (2.00000 2.00000)
3p4POINT (3.00000 3.00000)
points.plot()
# 创建多边形 GeoDataFramepolygons = gpd.GeoDataFrame(    [        {"id": "P1", "geometry": Polygon([(0, 0), (0, 2), (2, 2), (2, 0)])},        {"id": "P2", "geometry": Polygon([(1, 1), (1, 3), (3, 3), (3, 1)])}    ],    crs="EPSG:4326")polygons
idgeometry
0P1POLYGON ((0.00000 0.00000, 0.00000 2.00000, 2....
1P2POLYGON ((1.00000 1.00000, 1.00000 3.00000, 3....
polygons.plot()

sjoin函数

# 左连接join_left_df = points.sjoin(polygons, how="left")# 输出结果中的每一行都表示左侧GeoDataFrame中的一个几何对象与右侧GeoDataFrame中的一个几何对象进行了连接后得到的结果。# index_right表示右侧GeoDataFrame中的行索引# id_left:表示左侧GeoDataFrame中的几何对象的ID# id_right:表示右侧GeoDataFrame中的几何对象的ID# geometry:表示连接后的几何对象join_left_df
id_leftgeometryindex_rightid_right
0p1POINT (0.00000 0.00000)0P1
1p2POINT (1.00000 1.00000)0P1
1p2POINT (1.00000 1.00000)1P2
2p3POINT (2.00000 2.00000)0P1
2p3POINT (2.00000 2.00000)1P2
3p4POINT (3.00000 3.00000)1P2
# 右连接join_right_df = points.sjoin(polygons, how="right")join_right_df
index_leftid_leftid_rightgeometry
00p1P1POLYGON ((0.00000 0.00000, 0.00000 2.00000, 2....
01p2P1POLYGON ((0.00000 0.00000, 0.00000 2.00000, 2....
02p3P1POLYGON ((0.00000 0.00000, 0.00000 2.00000, 2....
11p2P2POLYGON ((1.00000 1.00000, 1.00000 3.00000, 3....
12p3P2POLYGON ((1.00000 1.00000, 1.00000 3.00000, 3....
13p4P2POLYGON ((1.00000 1.00000, 1.00000 3.00000, 3....
# 设置predicatejoin_right_within_df = points.sjoin(polygons, how="left", predicate="contains")join_right_within_df
id_leftgeometryindex_rightid_right
0p1POINT (0.00000 0.00000)NaNNaN
1p2POINT (1.00000 1.00000)NaNNaN
2p3POINT (2.00000 2.00000)NaNNaN
3p4POINT (3.00000 3.00000)NaNNaN
4.3 几何操作

GeoPandas提供了多种用于几何操作的函数,具体如下:

构造方法buffer(distance, resolution=16):返回一个GeoSeries,其中包含与每个几何对象距离在给定距离内的所有点的几何形状。boundary:返回一个GeoSeries,其中包含每个几何形状的集合理论边界的低维对象。centroid:返回一个GeoSeries,其中包含每个几何质心的点。convex_hull:返回一个GeoSeries,其中包含表示包含每个对象中所有点的最小凸多边形的几何形状,除非对象中的点数小于三个。对于两个点,凸包会折叠成一个线串;对于一个点,凸包是一个点。envelope:返回一个GeoSeries,其中包含包含每个对象的点或最小矩形多边形(其边与坐标轴平行)的几何形状。simplify(tolerance, preserve_topology=True):返回一个GeoSeries,其中包含每个对象的简化表示。在geopandas中,simplify函数可以用来简化多边形的形状,以减少地图数据的大小,同时也可以提高绘图的效率。当绘图数据特别大时,该函数很有用。tolerance:简化容差值,代表简化几何对象的形状后的最大允许误差。当 tolerance 值越小时,简化后的几何对象的形状越接近原始几何对象的形状。preserve_topology:是否保持拓扑结构,默认值为True,表示保持拓扑结构。unary_union:返回一个几何形状,其中包含GeoSeries中所有几何形状的联合。几何变化方法affine_transform(self, matrix):使用仿射变换矩阵来变换 GeoSeries 的几何形状。matrix 为一个包含6、12个元素的列表或元组(2d情况、3d情况)的仿射变换矩阵。关于 matrix 参数的使用需要有仿射变换的知识。rotate(ngle, origin="center", use_radians=False):旋转 GeoSeries 的坐标系。scale(xfact=1.0, yfact=1.0, zfact=1.0, origin="center"):沿着(x, y, z)三个方向缩放 GeoSeries 的几何形状。skew(xs=0.0, ys=0.0, origin="center", use_radians=False):基于原点origin,沿着 x 和 y 两个方向倾斜/扭曲 GeoSeries 的几何形状。translate(xoff=0.0, yoff=0.0, zoff=0.0):平移 GeoSeries 的坐标系。

构造方法使用示例

import geopandas as gpd# 加载数据集world = gpd.read_file(gpd.datasets.get_path("naturalearth_lowres"))# 展示结果world.plot()
# buffer函数buffered = world.geometry.buffer(distance=5)# 显示结果buffered.plot()
# 获取几何形状边界boundary = world.geometry.boundary# 显示结果boundary.plot()
# 获取几何质心centroids = world.geometry.centroid# 显示结果centroids.plot(marker="*", color="green", markersize=5)
# 获取几何形状的凸包convex_hulls = world.geometry.convex_hull# 显示结果convex_hulls.plot(alpha=0.5, edgecolor="k")
# 获取几何形状的外接矩形envelopes = world.geometry.envelope# 显示结果envelopes.plot(alpha=0.5, edgecolor="k")
# 对几何对象进行简化处理simplified = world.geometry.simplify(tolerance=0.1)# 显示结果simplified.plot(alpha=0.5, edgecolor="k")
merged = world.geometry.unary_union# 将合并后的几何对象转换为GeoDataFramegdf_merged = gpd.GeoDataFrame(geometry=[merged])# 打印后只有一行print(gdf_merged)gdf_merged.plot()
geometry0  MULTIPOLYGON (((-162.440 -79.281, -163.027 -78...

几何变化方法使用示例

# 读取数据集import geopandas as gpdnybb = gpd.read_file(gpd.datasets.get_path("nybb"))ax = nybb.plot()# 启用科学计数法ax.ticklabel_format(style="sci", axis="both", scilimits=(0,0))
from shapely.affinity import affine_transform# 仿射变换# 定义仿射变换参数a, b, d, e, xoff, yoff = 1.5, 0.5, 0, 0.5, 1.5, 0tmp = nybb.copy()# 对nybb数据集中的几何对象进行仿射变换tmp["geometry"] = tmp["geometry"].apply(lambda x: affine_transform(x, [a, b, d, e, xoff, yoff]))# 显示变换后的nybb数据集tmp.plot()
# 旋转nybb_rotate = nybb.geometry.rotate(angle=45)nybb_rotate.plot()
# 缩放nybb_scale = nybb.geometry.scale(xfact=2, yfact=2, zfact=1)nybb_scale.plot()
# 倾斜/扭曲nybb_skew = nybb.geometry.skew(xs=0.1, ys=0.2, use_radians=True)ax = nybb_skew.plot()# 启用科学计数法ax.ticklabel_format(style="sci", axis="both", scilimits=(0,0))
# 平移nybb_translated = nybb.geometry.translate(xoff=100000, yoff=100000, zoff=0)ax = nybb_translated.plot()# 启用科学计数法ax.ticklabel_format(style="sci", axis="both", scilimits=(0,0))
4.4 汇总

在geopandas中,dissolve函数可以对具有相同属性值的几何对象进行合并,从而生成新的几何对象。在汇总过程中,可以选择保留某些字段的信息,也可以对其他字段进行统计计算。dissolve函数如下:

geopandas.GeoDataFrame.dissolve(by=None, aggfunc="first", as_index=True, **kwargs)

函数参数介绍:

by: 可以是一个字段名,也可以是一列字段名的列表。表示按照哪些字段进行汇总。默认为None,即将所有要素合并成一个要素。aggfunc: 统计函数,用于对其他字段进行计算,可以是以下函数之一:"first": 返回第一个非空值。"last": 返回最后一个非空值。"mean": 返回平均值。"sum": 返回总和。"min": 返回最小值。"max": 返回最大值。自定义函数:可以传入自定义的聚合函数。as_index: 是否将by参数指定的字段作为行索引,默认为True*kwargs: 其他参数。

下面示例代码演示了dissolve函数的使用。

import geopandas as gpd# 读取湖北省地图数据data = gpd.read_file("https://geo.datav.aliyun.com/areas_v3/bound/420000_full.json")data.head()
adcodenamechildrenNumlevelparentsubFeatureIndexgeometry
0420100武汉市13city{"adcode": 420000}0MULTIPOLYGON (((113.71000 30.38892, 113.70961 ...
1420200黄石市6city{"adcode": 420000}1MULTIPOLYGON (((114.54626 30.06280, 114.54502 ...
2420300十堰市8city{"adcode": 420000}2MULTIPOLYGON (((111.04672 33.20292, 111.03242 ...
3420500宜昌市13city{"adcode": 420000}3MULTIPOLYGON (((112.07982 30.65932, 112.08643 ...
4420600襄阳市9city{"adcode": 420000}4MULTIPOLYGON (((111.58304 32.59654, 111.58514 ...
data.plot(cmap="tab20")
# 使用dissolve函数合并几何体,根据地级市的区县数分组dissolve_data = data.dissolve(by="childrenNum")dissolve_data.head()
geometryadcodenamelevelparentsubFeatureIndex
childrenNum
0MULTIPOLYGON (((113.02499 30.18293, 113.02826 ...429004仙桃市city{"adcode": 420000}13
3MULTIPOLYGON (((115.06176 30.26142, 115.05617 ...420700鄂州市city{"adcode": 420000}5
5MULTIPOLYGON (((112.06071 30.68840, 112.06988 ...420800荆门市city{"adcode": 420000}6
6MULTIPOLYGON (((114.94866 29.52531, 114.96668 ...420200黄石市city{"adcode": 420000}1
7MULTIPOLYGON (((113.43656 30.49471, 113.44782 ...420900孝感市city{"adcode": 420000}7
dissolve_data.plot(cmap="tab20")
# 使用dissolve函数合并几何体,根据地级市的区县数分组,其他列求均值dissolve_data = data.dissolve(by="childrenNum", aggfunc="mean")dissolve_data.head()
geometryadcodesubFeatureIndex
childrenNum
0MULTIPOLYGON (((113.02499 30.18293, 113.02826 ...429009.014.5
3MULTIPOLYGON (((115.06176 30.26142, 115.05617 ...421000.08.0
5MULTIPOLYGON (((112.06071 30.68840, 112.06988 ...420800.06.0
6MULTIPOLYGON (((114.94866 29.52531, 114.96668 ...420700.05.5
7MULTIPOLYGON (((113.43656 30.49471, 113.44782 ...420900.07.0
4.5 缺失值与空值处理

在使用geopandas处理地理空间数据时,经常会遇到NoneEmpty这两个概念。虽然它们都表示缺失值,但它们之间有着一些区别。

None:表示属性或者列的值不存在,或者没有被填充。在geopandas中,如果一个geometry列的值为None,那意味着这个几何对象不存在。Empty:表示属性或者列的值存在,但是值为空。在geopandas中,如果一个geometry列的值为空,那意味着这个几何对象是存在的,但是它没有任何形状或者坐标信息。

以下为具有一个多边形、一个缺失值和一个空多边形的GeoSeries示例:

from shapely.geometry import Polygons = gpd.GeoSeries([Polygon([(0, 0), (1, 1), (0, 1)]), None, Polygon([])])s
0    POLYGON ((0.00000 0.00000, 1.00000 1.00000, 0....1                                                 None2                                        POLYGON EMPTYdtype: geometry

在geopandas空间运算中,缺失的几何图形通常会传播。在结果中,这些缺失的几何图形也会缺失。另一方面,空的几何图形被视为几何图形。结果将取决于所进行的运算。如下所示:

s.area
0    0.51    NaN2    0.0dtype: float64

我们可以通过isna函数和is_empty属性判断是否为缺失值或者空值:

# 判断缺失值s.isna()
0    False1     True2    Falsedtype: bool
# 判断空值s.is_empty
0    False1    False2     Truedtype: bool
# 判断缺失或为空s.is_empty | s.isna()
0    False1     True2     Truedtype: bool
# 提取既不缺失也不为空的值s[~(s.is_empty | s.isna())]
0    POLYGON ((0.00000 0.00000, 1.00000 1.00000, 0....dtype: geometry
5 背景地图叠加

contextily是一个Python库,它提供了一种简单的方法将背景地图(通常是Web瓦片地图,如OpenStreetMap、Stamen Maps、Mapbox等)添加到地理空间数据可视化中。使用contextily库可以使地理空间数据可视化更加生动、直观,同时可以提供更多的地理信息。瓦片地图是一种基于网格的地图显示方式,将地图划分为多个小块,每个小块称为“瓦片”,每个瓦片都有自己的坐标和编号。这些瓦片可以按需加载,使用户能够快速地浏览地图,同时减少了加载时间和资源消耗。瓦片地图常用于在线地图应用程序,例如谷歌地图和百度地图。contextily支持使用WGS84 (EPSG:4326)和Spheric Mercator (EPSG:3857)坐标系,在Web地图应用程序中,一般使用EPSG:3857(以米为单位)来显示瓦片地图,并使用EPSG:4326(以经纬度为单位)来标记瓦片地图上的位置。

contextily库的主要功能包括:

从Web地图提供商获取地图图层将地图图层与地理空间数据集合并使用Matplotlib或Bokeh绘制地图

本文主要介绍contextily简单使用,contextily具体使用可参考其官方文档:contextily-doc。contextily库中基于add_basemap函数在地图上添加背景地图。下面是该函数常用可用参数的介绍:

ax: matplotlib axes对象,用于绘制地图crs: 输出地图的坐标系,默认为"EPSG:3857"source: 底图的来源,支持多种来源,如OpenStreetMap、Stamen Terrain、Stamen Toner等等,默认为OpenStreetMapzoom: 底图的缩放级别,默认为None,自动根据ax的extent和crs计算。zoom值越高,底图的缩放级别就越大,地图显示的范围也就越小,细节也会越来越清晰。url: 底图的url地址,默认为None,自动根据source和zoom计算。attribution: 底图的版权信息,默认为Nonealpha: 底图的透明度,默认为1.0*kwargs: 其他matplotlib.image()函数的可选参数,如cmapvminvmax等等

⚠⚠source参数选择不同底图的来源,可能需要大量时间或者特定网络,如果失败多重试运行代码。

5.1 简单背景地图叠加
import geopandas as gpd# 读取深圳市地图数据data = gpd.read_file("https://geo.datav.aliyun.com/areas_v3/bound/440300_full.json")# 简单绘图ax = data.plot(alpha=0.5, edgecolor="k")
# 确定数据所使用的坐标系data.crs# 将数据集所使用坐标系转为EPSG:3857data_wm = data.to_crs(epsg=3857)
import contextily as cximport matplotlib.pyplot as plt fig, ax = plt.subplots(figsize=(10, 10))ax = data_wm.plot(ax = ax, alpha=0.5, edgecolor="k")# 将自动下载瓦片地图cx.add_basemap(ax)# 保存地图fig.savefig("save.jpg", pad_inches=0.5, bbox_inches="tight", dpi=300)

在上面的代码中,如果仅使用经纬度数据叠加瓦片地图,需要在add_basemap函数中设置crs参数,如下所示:

import contextily as cximport matplotlib.pyplot as plt fig, ax = plt.subplots(figsize=(10, 10))ax = data.plot(ax = ax, alpha=0.5, edgecolor="k")# 将自动下载瓦片地图cx.add_basemap(ax, crs=data.crs)

可以通过调整zoom参数改变背景瓦片地图的细节程度,建议zoom值不要过大,下载速度太慢。此外可以通过设置attribution=""去除绘图水印。

ax = data_wm.plot(figsize=(10, 10), alpha=0.5, edgecolor="k")cx.add_basemap(ax, zoom=12, attribution="")
5.2 定制化背景地图

通过设置add_basemap的source参数能够指定不同的数据源,以在地图上添加不同类型的底图。如下所示:

ax = data_wm.plot(figsize=(10, 10), alpha=0.5, edgecolor="k")cx.add_basemap(ax, source=cx.providers.Stamen.TonerLite)ax.set_axis_off()

当然也可以叠加多个背景地图,如下所示:

ax = data_wm.plot(figsize=(10, 10), alpha=0.5, edgecolor="k")cx.add_basemap(ax, source=cx.providers.Stamen.TonerLite)cx.add_basemap(ax, source=cx.providers.Stamen.TonerLabels)

此外也可以叠加多个不同来源的背景图层。如下所示:

ax = data_wm.plot(figsize=(10, 10), alpha=0.5, edgecolor="k")cx.add_basemap(ax, source=cx.providers.Stamen.Watercolor, zoom=12)cx.add_basemap(ax, source=cx.providers.Stamen.TonerLabels, zoom=10)

从上面的案例我们可以看到,contextily通过Provider预置提供商的名称来获取相应的Web瓦片地图。contextily所有预置的地图提供商可以通过以下cx.providers命令获取。可以尝试根据这些提供商定制瓦片地图格式。

# cx.providers

除了使用contextily预置的地图提供商,可以通过source设置给定瓦片地图地址来指定需要添加的底图。例如可以添加天地图,高德地图,腾讯地图的瓦片地图的地址。一些示例的瓦片地图地址可见:高德谷歌腾讯天地图地图瓦片url和在geopandas中叠加在线地图。一般地图服务提供XYZ瓦片地图链接,其中的xyz代表了地图的坐标系。如下所示:

x:表示在地图水平方向上的位置,从左到右递增,即经度值。y:表示在地图竖直方向上的位置,从上到下递增,即纬度值。z:表示地图的缩放级别,从0开始递增,数值越大,地图显示的范围越小,细节越丰富。

在瓦片地图中,地图被分成了许多小块,每个小块都有一个唯一的编号,也就是xyz坐标系。当我们使用地图服务时,通过改变xyz的值,就可以获取到不同位置、不同缩放级别下的地图瓦片,从而达到展示不同地图的目的。直接通过url设置瓦片地图示例如下:

fig, ax = plt.subplots(figsize=(10, 10))ax = data_wm.plot(ax=ax, alpha=0.5, edgecolor="k")cx.add_basemap(ax,                 source="https://webst01.is.autonavi.com/appmaptile?style=6&x={x}&y={y}&z={z}",                zoom=12)fig.savefig("save.jpg", pad_inches=0, bbox_inches="tight", dpi=300)
fig, ax = plt.subplots(figsize=(10, 10))ax = data_wm.plot(ax=ax, alpha=0.5, edgecolor="k")cx.add_basemap(ax,                 source="https://webrd02.is.autonavi.com/appmaptile?lang=zh_cn&size=1&scale=1&style=8&x={x}&y={y}&z={z}",                zoom=12)fig.savefig("save.jpg", pad_inches=0, bbox_inches="tight", dpi=300)ax.set_xlim(data_wm.total_bounds[0], data_wm.total_bounds[2])ax.set_ylim(data_wm.total_bounds[1], data_wm.total_bounds[3])
(2559177.946084248, 2615308.057854809)
5.3 离线背景地图

在有些时候我们需要离线使用背景瓦片地图,contextly提供bounds2raster函数用于根据给定的空间范围和地图缩放级别,将在线地图服务中的栅格数据下载为本地文件。bounds2raster函数常用参数如下:

w:float类型,表示空间范围的最小值。s:float类型,表示空间范围的最小值。e:float类型,表示空间范围的最大值。n:float类型,表示空间范围的最大值。path:str类型,表示下载的栅格数据文件的保存路径。zoom:int或者字符串类型,表示地图缩放级别。如果为字符串类型,可以设置为"auto",表示自动确定最佳的缩放级别。source:str类型,表示地图服务的地址。ll:bool类型,表示w、s、e、n是否使用经纬度坐标系,默认为False。wait:int类型,表示两次下载之间的等待时间,单位为秒。默认为0。max_retries:int类型,表示下载失败后最大的重试次数。默认为2次。

bounds2raster函数返回RGB图像数组和瓦片图像边界框[minX,maxX,minY,maxY],此外由于网络地图总是基于WGS84 Web Mercator(EPSG:3857)坐标系,因此bounds2raster函数返回和保存的图片都是基于EPSG:3857坐标系。

import geopandas as gpd# 读取郑州市地图数据data = gpd.read_file("https://geo.datav.aliyun.com/areas_v3/bound/410100_full.json")# 简单绘图ax = data.plot(alpha=0.5, edgecolor="k")
# 叠加地图ax = data.plot(alpha=0.5, edgecolor="k")# crs告诉数据集用的坐标系统,这里data.crs为WGS 84(经纬度)cx.add_basemap(ax,                crs=data.crs,                source=cx.providers.Stamen.Watercolor               )

提取待绘图区域的边界信息

west, south, east, north = bbox = data.total_boundsbbox
array([112.721178,  34.262109, 114.220962,  34.989506])

根据边界信息下载数据

import contextily as cximport matplotlib.pyplot as plt img, ext = cx.bounds2raster(west,                             south,                             east,                             north,                             "demo.tif",                             source=cx.providers.Stamen.Watercolor,                             ll=True                            )
# 展示下载的数据plt.axis("off")plt.imshow(img)# 边界范围ext
(12523442.714243276, 12719121.506653327, 4030983.1236470537, 4187526.157575096)

有了背景地图,add_basemap中的source函数设置文件路径地址就可以离线叠加地图。

ax = data.plot(alpha=0.5, edgecolor="k")# crs告诉数据集用的坐标系统,这里data.crs为WGS 84(经纬度)cx.add_basemap(ax,                crs=data.crs,                source="demo.tif",               )
6 参考GeoPandasGeoPandas-docGeoPandas Examples GalleryPython绘制数据地图1-GeoPandas入门指北Python绘制数据地图2-GeoPandas地图可视化matplotlib-scalebarcontextilycontextily-doc高德谷歌腾讯天地图地图瓦片url在geopandas中叠加在线地图

推荐内容