利用Pyhton提取宗地最高楼层
利用arcpy来处理数据,并用geopandas来出图,这次的目标是提取宗地内楼层的最高楼层数。
data:image/s3,"s3://crabby-images/492a0/492a0482222e07fbd2536bad559eabb883ea7efd" alt=""
预处理
import geopandas as gpd
import arcpy
from arcpy import env
import pandas as pd
import matplotlib.pyplot as plt
import numpy as np
import matplotlib.patches as mpatches
from mpl_toolkits.basemap import Basemap
from matplotlib_scalebar.scalebar import ScaleBar
plt.rcParams['font.family'] = ['sans-serif']
plt.rcParams['font.sans-serif'] = ['SimHei']# 替换sans-serif字体为黑体
plt.rcParams['axes.unicode_minus'] = False # 解决坐标轴负数的负号显示问题
path = r'F:\ArcGIS\ArcGIS文件\ModelBuilder\data\chp05'
env.workspace = path
par_path = path+'\宗地.shp'
floor_path = path + '\楼层数.shp'
arcpy处理
## 定义投影
## 都建议大家这么写,有异常捕捉
try:
# set local variables
in_dataset = par_path #"forest.shp"
# get the coordinate system by describing a feature class
dsc = arcpy.Describe(floor_path)
coord_sys = dsc.spatialReference
# run the tool
arcpy.DefineProjection_management(in_dataset, coord_sys)
# print messages when the tool runs successfully
print(arcpy.GetMessages(0))
except arcpy.ExecuteError:
print(arcpy.GetMessages(2))
except Exception as ex:
print(ex.args[0])
data:image/s3,"s3://crabby-images/e8638/e8638fc0e17be03be670decd3ea1fbe4de464d8e" alt=""
path = r'F:\ArcGIS\ArcGIS文件\ModelBuilder\data\chp05'
par_path = path+'\宗地.shp'
floor_path = path + '\楼层数.shp'
floor = gpd.GeoDataFrame.from_file(floor_path)
floor.plot()
data:image/s3,"s3://crabby-images/eaeac/eaeac616e4b45fcf0babb13c329a0baf8232d6b2" alt=""
parcels = gpd.GeoDataFrame.from_file('F:\ArcGIS\ArcGIS文件\ModelBuilder\data\chp05\宗地.shp')
parcels.plot()
data:image/s3,"s3://crabby-images/3cb58/3cb58644a3fab95ce79d965824eef4af7785f913" alt=""
## 如何叠图,关键ax = ax
fig, ax = plt.subplots(figsize=(8, 8))
floor['coords'] = floor['geometry'].apply(lambda x: x.representative_point().coords[0])
for n, i in enumerate(floor['coords']):
plt.text(i[0], i[1], floor['floor'][n], size=20)
parcels.plot(ax=ax,alpha=0.3, edgecolor='k',label='宗地')
floor.plot(ax=ax,label='大楼')
#plt.savefig('待处理.png',dpi=300)
## 就是开头的图片,这里就不演示了
数据处理和清洗
## arcpy获取字段
featureclass = par_path
field_names = [f.name for f in arcpy.ListFields(featureclass)]
field_names
data:image/s3,"s3://crabby-images/b77ed/b77eda60f480794c9da056175af56244488b023b" alt=""
## 删除文件,我做过一遍的
in_data = 'F:\ArcGIS\ArcGIS文件\ModelBuilder\data\chp05\楼层数_SpatialJoin.shp'
try:
#arcpy.Delete_management (in_data)
# print messages when the tool runs successfully
print(arcpy.GetMessages(0))
except arcpy.ExecuteError:
print(arcpy.GetMessages(2))
except Exception as ex:
print(ex.args[0])
data:image/s3,"s3://crabby-images/d2c89/d2c89f7a91d72bc5adfa8e4250cdec7bec9261da" alt=""
## 空间连接
try:
# set local variables
target_features = 'F:\ArcGIS\ArcGIS文件\ModelBuilder\data\chp05\楼层数.shp'
join_features = 'F:\ArcGIS\ArcGIS文件\ModelBuilder\data\chp05\宗地.shp'
out_feature_class = 'F:\ArcGIS\ArcGIS文件\ModelBuilder\data\chp05\楼层数_SpatialJoin.shp'
field_mapping = ['floor','宗地号']
# run the tool
arcpy.SpatialJoin_analysis(target_features, join_features, out_feature_class,)
# print messages when the tool runs successfully
print(arcpy.GetMessages(0))
except arcpy.ExecuteError:
print(arcpy.GetMessages(2))
except Exception as ex:
print(ex.args[0])
data:image/s3,"s3://crabby-images/b9322/b9322aa60a295de909e617793da3b41a79753bc6" alt=""
## 删除字段
f_Spat = 'F:\ArcGIS\ArcGIS文件\ModelBuilder\data\chp05\楼层数_SpatialJoin.shp'
try:
featureclass = 'F:\ArcGIS\ArcGIS文件\ModelBuilder\data\chp05\楼层数_SpatialJoin.shp'
field_names = [f.name for f in arcpy.ListFields(featureclass)]
field_names
del_features = ['Text','街坊号','宗地预','Sum_占地','Sum_建筑','土地用','登记号','权利人',
'宗地类','土地坐','面积','土地证','权属性','使用权','宗地四','宗地面']
arcpy.DeleteField_management(f_Spat,del_features )
# print messages when the tool runs successfully
print(arcpy.GetMessages(0))
except arcpy.ExecuteError:
print(arcpy.GetMessages(2))
except Exception as ex:
print(ex.args[0])
## 数据最后不要相互干扰
f_num_path = 'F:\ArcGIS\ArcGIS文件\ModelBuilder\data\chp05\楼层数_SpatialJoin.shp'
f_num_path
f_num = gpd.GeoDataFrame.from_file(f_num_path)
f_num.plot()
data:image/s3,"s3://crabby-images/6a6cb/6a6cb0661b485836e82d72f16d780ddf4751ce8c" alt=""
f_num.head()
data:image/s3,"s3://crabby-images/f6679/f66791da9d80b062ecca42febd2f212395ed3dfc" alt=""
f_num.shape
data:image/s3,"s3://crabby-images/3537a/3537a197bd4cd3206796d8ba699866eecfd6bfff" alt=""
表面上没有什么变化,但是结果属性表多了宗地号,还有其他两个字段,接下来可视化一下。
fig, ax = plt.subplots(figsize=(8, 8))
f_num['coords'] = f_num['geometry'].apply(lambda x: x.representative_point().coords[0])
ax = f_num.plot(ax=ax,column='floor', legend=True, s =f_num['floor']*20, cmap='Reds', edgecolor='k')
for n, i in enumerate(f_num['coords']):
plt.text(i[0], i[1], f_num['floor'][n], size=20)
parcels.plot(ax = ax, alpha =0.3,edgecolor = 'k')
plt.title('宗地楼层分布图', size=25)
plt.grid(True, alpha=0.3)
plt.savefig('宗地楼层分布图.png',dpi=300)
data:image/s3,"s3://crabby-images/53874/538740ad3f1f1b7bf48943576495b6d8f4a0091b" alt=""
数据透视和分组计算
## 其实我是想要分组计算的结果,就是大楼要有相对应的坐标
max_floor = pd.pivot_table(f_num,index=['宗地号',],values=['floor','geometry'],aggfunc=np.max)
max_floor.head()
data:image/s3,"s3://crabby-images/08054/080543db0ebb6899db833549f2fc9488632f7c01" alt=""
f_num_sort = f_num
f_num_sort = f_num.sort_values('floor', ascending=False).groupby('宗地号', as_index=False).first()
f_num_sort = f_num_sort.drop(labels=['Join_Count','TARGET_FID'],axis=1) # axis=1 表示按列删除,删除gender、age列
f_num_sort.head()
data:image/s3,"s3://crabby-images/3e421/3e4212b08cc91e9e92836879001708d15bde9707" alt=""
data:image/s3,"s3://crabby-images/793ba/793bab8d8d9c9fd01cbe8b6be71aa9513a97662b" alt=""
其实我是想要分组计算的结果,就是大楼要有相对应的坐标,有谁知道数据透视怎么得到这个分组计算的结果,不吝赐教。
数据连接
result = pd.merge(max_floor,parcels,on='宗地号')
result.head()
data:image/s3,"s3://crabby-images/36d95/36d956333fbacbaa3c0a678ff74849d1a97a01b1" alt=""
## 删除不需要的字段
result = result[['宗地号','floor','geometry']]
result = gpd.GeoDataFrame(result) ## 保证数据是GeoDataFrame格式
result.head()
data:image/s3,"s3://crabby-images/3698d/3698d2356901bf6c8183d9ddf27035a89ac846f3" alt=""
出图
f_num_sort = gpd.GeoDataFrame(f_num_sort)
f_num_sort.plot()
data:image/s3,"s3://crabby-images/5b961/5b9613e1ba0104c6275381bbf629a079973c2468" alt=""
## 推荐写法,变成一个实例来写
fig, ax = plt.subplots(figsize=(8,8))
result['coords'] = result['geometry'].apply(lambda x: x.representative_point().coords[0])
ax = result.plot(ax=ax,column='floor',cmap='Reds',edgecolor='k',legend=True)
for n, i in enumerate(result['coords']):
ax.text(i[0],i[1], result['floor'][n],size=20)
f_num_sort.plot(ax=ax,alpha=0.8)
ax.set_title('宗地最高楼层分布图',size=25)
ax.grid(True,alpha=0.3)
plt.savefig('效果图.png',dpi=300)
data:image/s3,"s3://crabby-images/52a0f/52a0f1298abebd274e4314acc2d6f30650ed8eff" alt=""
## 制作标准地图
fig, ax = plt.subplots(figsize=(8,8))
result['coords'] = result['geometry'].apply(lambda x: x.representative_point().coords[0])
ax = result.plot(ax=ax,column='floor',cmap='Reds',edgecolor='k',legend=True)
for n, i in enumerate(result['coords']):
ax.text(i[0],i[1], result['floor'][n],size=20)
f_num_sort.plot(ax=ax,alpha=0.8)
ax.set_title('宗地最高楼层分布图',size=25)
ax.grid(True,alpha=0.3)
# 添加比例尺
scalebar = ScaleBar(dx=1*10**-3,units='km',length_fraction=0.1,
font_properties={'family': 'Times New Roman', 'weight': 'normal', 'size': 12},
location=3,sep=0.5,frameon=False) ## location参数选择放的位置
ax.add_artist(scalebar)
# 添加指北针
x, y, arrow_length = 0.9, 0.92, 0.07
ax.annotate('N', xy=(x, y), xytext=(x, y-arrow_length),
arrowprops=dict(facecolor='black', width=4, headwidth=7),
ha='center', va='top', fontsize=10,
xycoords=ax.transAxes)
plt.savefig('宗地最高楼层分布图.png',dpi=300)
data:image/s3,"s3://crabby-images/92b08/92b08c186ea7ea1b1621486252b5e006ce6b0b21" alt=""
总结
arcpy是处理地理数据的好帮手,而geopandas和matplotlib有这强大的数据可视化能力,两者结合起来,可以创造很多的可能性,数据一定要正确,投影和拓扑都要正确。还有就是异常捕捉,这个必不可少,我们代码写多了,一点要异常捕捉,多输出看看。欢迎大家转发和关注。
网友评论