地理信息可视化是我们测绘、气象、地球科学方向的学子常会面对的任务。一副好图胜过大段的文字,可以直观揭示事物的特点,帮助我们探索问题的本质。高质量的成图也可以提高我们科研论文和工作报告的表现力,提高文章的档次。
表1中对笔者和周围朋友常用的
地球科学绘图软件
做了简单的汇总和对比。一般来说,Excel成图美观度较差,难以通过脚本完成自动化操作,主要用于快速的数据分析。因此,用于地理信息可视化的软件主要是GMT、Matplotlib的basemap模块,和Matlab的M_map模块。
GMT和M_map笔者早先使用过,目前笔者使用的是Python的basemap模块。除了功能强大,语法简单、清晰,轻量级(安装包占用空间不大)等优势之外,另一个主要原因在于Python强大的生态和活跃的社区。无论是数据处理、分析,还是科学计算,日常工作自动化等方面,Python都有丰富且热门的模块。笔者使用Python作为主要开发语言之一,因此就选择了basemap模块,将其集成在数据处理和分析任务中。
笔者周围不少小伙伴对basemap画图很感兴趣。本文根据小伙伴们对basemap模块关心的一些问题,收集资料,结合笔者的一些初步经验,对basemap画图做了初步总结,供大家参考。
文章全文约6000字,包括15个代码(片段),21幅样例图。受篇幅所限,部分较长的代码并未在文中全部贴出。如果您有兴趣,可在后台留言"Python-basemap"和您的邮箱地址,笔者会将本期PDF文稿和相应数据、脚本发送给您。
在数据可视化过程中,我们需要将数据在地图上画出来。 比如说我们在地图上画出城市人口,飞机航线,军事基地,矿藏分布等等。这样的地理绘图有助于读者理解空间相关的信息。basemap 是Python的一个强大的负责实现地理信息可视化的库,是Matplotlib的一个附加工具包,通过结合 matplotlib 可以绘制出很多漂亮的地图。
basemap包括GSSH海岸线数据集,以及来自GMT的河流、州和国家边界的数据集。这些数据集可用于在地图上以几种不同的分辨率绘制海岸线,河流和政治边界。basemap底层使用了Geometry Engine-Open Source(GEOS)库,用来将海岸线和边界特征剪切到所需的地图投影区域。此外,basemap还提供读取shapefile的功能。
basemap面向地球科学家的需求,特别是海洋学家和气象学家。起初,杰夫·惠特克(Jeff Whitaker)写“底图”(basemap)用来帮助他进行气候和天气预报的研究。到现在,basemap 不断被开发扩展已经具备了很多功能。多年来,basemap的功能随着各个学科(如生物学,地质学和地球物理学)的科学家的要求和贡献的新功能而演变。
3 关于安装
问1:basemap模块的大小和安装的难易程度?
答:basemap模块约为120M左右,其依赖库pyproj约为20M左右。
由于basemap还没有pip检索,因此传统的(pip install basemap)安装方法不适用。
Windows用户需要到
https://www.
lfd.uci.edu/~gohlke/pyt
honlibs/
下载对应的wheel文件到本地,然后控制台进入其所在目录,使用pip install xxxx.whl文件安装。其依赖于pyproj库。具体安装过程参考
https://
zhuanlan.zhihu.com/p/34
509847
。Linux用户安装步骤可参考
https://
matplotlib.org/basemap/
users/installing.html
,包括下载源码、安装GEOS库、设置GEOS_DIR环境变量等步骤,安装完成后可进入到examples文件夹,运行测试例子。 此外还可以通过Anaconda的Python发行版安装。
建议同时安装Pillow库, 可用于以更多不同格式保存图片文件,以及调用
bluemarble()
,
etopo()
,
shadedrelief()
and
warpimage()
函数。
问2:basemap使用是否方便?有没有相关介绍或例子?
答:basemap基于GEOS的地图二维数据,其底图数据库与GMT相同,封装了大量常用的地图投影、坐标转换功能,利用简洁的Python语法支持绘出多种多样的地理地图。
如下所示,basemap的功能有且不限于绘制地图、等高线、降水量图、海平面气压天气图、晨昏线等,具体功能及对应实例代码可参考
https://
matplotlib.org/basemap/
users/examples.html
。本文后续内容也给出了多个样例。
4 基本操作
问3:能否在地图上画线、画箭头之类的标注线?
答:可以。比如需要连接地图上已知经纬度(lat1,lon1;lat2,lon2)的两个点,可使用plot来进行连线,basemap中暂没有画箭头的函数,可用plt.arrow绘制箭头。一些更高级的用法,比如画球面距离,可使用drawgreatcircle函数。下图实例为绘制伦敦到纽约的直线以及球面距离。
1 from mpl_toolkits.basemap import Basemap
2 import matplotlib.pyplot as plt
3 import numpy as np
5 fig=plt.figure()
6 ax=fig.add_axes([0.1,0.1,0.8,0.8])
7 mymap = Basemap(llcrnrlon=-100.,llcrnrlat=20.,urcrnrlon=20.,urcrnrlat=60.,\
8 rsphere=(6378137.00,6356752.3142),\
9 resolution='l',projection='merc',\
10 lat_0=40.,lon_0=-20.,lat_ts=20.)
11 # nylat, nylon are lat/lon of New York
12 nylat = 40.78; nylon = -73.98
13 # lonlat, lonlon are lat/lon of London.
14 lonlat = 51.53; lonlon = 0.08
15 mymap.drawgreatcircle(nylon,nylat,lonlon,lonlat,linewidth=2,color='b')
16 mymap.plot([nylon,lonlon],[nylat,lonlat],linewidth=2,color='r',latlon='True')
17 mymap.drawcoastlines()
18 mymap.fillcontinents()
19 mymap.drawparallels(np.arange(10,90,20),labels=[1,1,0,1])
20 mymap.drawmeridians(np.arange(-180,180,30),labels=[1,1,0,1])
21 plt.show()
问4:如何导出矢量图?
答:使用函数savefig(),保存矢量图只需要三个参数,即文件名fname,DPI和文件格式format。比如保存文件名为“vectorgraph”,dpi为600,eps格式的图形,可使用如下代码。目前savefig支持的格式包括:eps,jpeg,jpg,pdf,pgf,png,ps,raw,rgba,svg,svgz,tif,tiff。值得注意的是,如果文件名含后缀,则format参数可省略。
1 fig.savefig('vectorgraph',dpi=600,format='eps')
问5:经纬度怎么更简洁的表示,全部显示有点多和乱,国际上有惯例吗?
答:经纬度的表示没有国际惯例,需要用户自己设置。drawparallels和drawmeridians函数可通过设置经纬度间隔来显示图上的经纬度。且能分别设置图片左右边框和上下边框是否显示经纬度。
1 parallels=np.arange(-90.,90.001.,30.)#范围[-90,90]间隔为30
2 mymap.drawparallels(parallels,labels=[False,True,True,False])
3 meridians=np.arange(-180.,180.001.,60.)#范围[-180,180]间隔为60
4 mymap.drawmeridians(meridians,labels=[True,False,False,True])
问6:地图怎么画比例?
答:画比例尺的函数为drawmapscale。下图给出了两种比例尺示例。
1 from mpl_toolkits.basemap import Basemap
2 import matplotlib.pyplot as plt
3 import numpy as np
5 plt.figure(figsize=(6, 6))
7 mymap = Basemap(llcrnrlon=-10,llcrnrlat=35, urcrnrlon=5.,urcrnrlat=45.,resolution='i', projection='merc', lat_0 = 39.5, lon_0 = -3.25)
10 mymap.fillcontinents(color='gray', lake_color='lightskyblue')
11 mymap.drawcoastlines()
12 mymap.drawmapboundary(fill_color='skyblue')
14 mymap.drawmeridians(np.arange(-10, 5 + 0.001, 5), labels=[1, 1, 1, 1])
15 mymap.drawparallels(np.arange(35, 45 + 0.001, 5), labels=[1, 1, 1, 1])
17 mymap.drawmapscale(-4., 36.0, 0.25, 39.5, 500, barstyle='fancy')
18 mymap.drawmapscale(2., 36.0, 4.25, 39.5, 500, fontsize = 10)
20 plt.savefig('mapscale.png', dpi=360)
21 plt.show()
问7:能否调节地图分辨率?
答:可以在创建 basemap 图像时设置所需的分辨率。
basemap
类的
resolution
参数设置分辨率级别,他们是
'c'
(原始),
'l'
(低),
'i'
(中),
'h'
(高),
'f'
(完整)或
None
(如果没有使用边界)。这个选项很重要,在全局地图上设置高分辨率边界可能非常慢。
问8:如何设置投影方式?
答:basemap的投影方式是由属性projection控制,提供了包括:Robinson 罗宾逊投影、Orthographic 正射投影、Mercator 魔卡托投影、Sinusoidal 正弦投影、Lambert 兰勃特投影在内的近30种投影方式。默认投影为‘cyl’(圆柱投影)。具体的投影方式对应的设置方法可参考
https://matplotlib.org/basemap
1 plt.figure(figsize(8,6))
2 mymap=Basemap(projection='ortho',lat_0=0,lon_0=0)#设置投影方式及投影中心
问9:地图能否设置长宽比?
答:basemap地图的长宽比是固定的。例如,robin投影为椭圆,不能通过设置进行拉伸;cyl投影为矩形,长宽为固定比例,也不能改变。
问10:basemap能否显示省级信息?
答:全球的行政区划数据都可以在GADM下载(
https://
gadm.org/download_count
ry_v3.html
)。
如果想在basemap中显示中国区域的省级信息,需到上述网址下载中国区域的shapefile文件,解压之后的文件名中会有0,1,2,3后缀,分别对应于第零、一、二、三级行政区划界限,基本相当于中国大陆地区的边界、省界、市界和区县界。可通过readshapefile函数读取相关信息。(PS:数据中有领土纠纷的部分不代表笔者立场)
问11:假如数据是离散的,是怎么样的插值让图看起来不是单个网格?
答:如果有固定的数学模型,可根据数学模型加密数据,使图形更平滑。如果没有确定的数学模型,python中提供了一种样条插值办法,可使用scipy.interpolate模块下的interpld函数实现插值,加密数据。
问12:basemap画图有办法用滚轮进行缩放吗?
答:可以。缩放等鼠标交互绘制操作不属于basemap库,可使用fig.canvas.mpl_connect()函数来绑定相关fig的交互事件,此类交互事件包括:鼠标按下与释放,按键按下与释放,滚轮滚动等,其中滚轮操作属性为“button”,滚动属性为“up”和“down”。具体可参考
https://www.
jianshu.com/p/cf205a759
问13:能否绘制多个子图在一个图中,比如卫星弧段图,SNR图等?
答:Python可通过subplot函数画多个子图。调用形式如:subplot(nrows,ncols,index),图表的整个绘图区域被分成nrows行和ncols列,按照从左到右,从上到下的顺序对每个子区域进行编号,左上的子区域编号为1。index参数指定创建的Axes对象所在的区域。
matplotlib官网上的一个2*2多子图例子如下所示:
1 import matplotlib.pyplot as plt
2 import numpy as np
4 # Some example data to display
5 x = np.linspace(0, 2 * np.pi, 400)
6 y = np.sin(x ** 2)
8 fig, ((ax1, ax2), (ax3, ax4)) = plt.subplots(2, 2)
9 fig.suptitle('Sharing x per column, y per row')
10 ax1.plot(x, y)
11 ax2.plot(x, y**2, 'tab:orange')
12 ax3.plot(x, -y, 'tab:green')
13 ax4.plot(x, -y**2, 'tab:red')
15 for ax in fig.get_axes():
16 ax.label_outer()
比如要将卫星弧段图和SNR图两个图按照上下顺序放置于同一张图时,可采用如下方式:
1 import matplotlib.pyplot as plt
3 plt.figure()
4 plt.subplot(211)
5 plt.plot(t1, Satarc)
7 plt.subplot(212)
8 plt.plot(t2, SNR)
5 高级操作
问14.1:basemap可以实现画全球和局部测站分布图吗?
问14.2:测站分布图可以按照可观测卫星系统类型进行站点分类吗?
答:可以实现。对于全球或局部测站分布图,可通过控制经纬度显示范围得到。对于想表示的要素,比如卫星系统或者DOP值,需要将测站能观测到的卫星系统或者DOP值作为参数读取到代码中,用不同颜色对参数的值进行表示即可。以下将以按照可观测卫星系统的值给出全球测站分布图。标注测站ID,需要在代码中进行微调,将重叠部分的ID上下左右移动。
1 from mpl_toolkits.basemap import Basemap
2 import matplotlib.pyplot as plt
3 import numpy as np
5 readfile()#读取测站坐标文件存放于列表 sites lons lats N_G N_E N_C N_J
6 plt.figure(figsize=(12,9))
7 mymap = Basemap(projection='robin', lat_0=0, lon_0=0,resolution='i',
8 area_thresh=5000.0)
9 mymap.fillcontinents(color='white', lake_color='lightskyblue')
10 mymap.drawmapboundary(fill_color='skyblue')
11 mymap.drawmeridians(np.arange(0, 360, 60), labels=[1,0,0,1])
12 mymap.drawparallels(np.arange(-90, 90.001, 30), labels=[1,0,0,1])
13 x, y = mymap(lons, lats)
15 for name,lon,lat,n_G,n_E,n_J,n_C in zip(sites,x,y,N_G,N_E,N_J,N_C):
16 print (name, lon, lat,n_G,n_E,n_J,n_C)
17 if n_J>0:#可设置不同标记,大小和颜色,此处可根据需要替换为DOP值等
18 plt.plot(lon, lat, marker='o', color='red', markersize=9)
19 if n_C>0:
20 plt.plot(lon, lat, marker='o', color='blue', markersize=7)
21 if n_E>0:
22 plt.plot(lon, lat, marker='o', color='green', markersize=5)
23 if n_G>0:
24 plt.plot(lon, lat, marker='o', color='yellow', markersize=3)
25 plt.legend(loc=0)
26 plt.show()
问15:basemap画全球气温分布图,该怎么在底图上叠加数据?
答:在绘制气温分布图时,需要用到读写netcdf文件的第三方库:netCDF4。对于标准的气温数据,可在NCEP/NCAR上下载.nc文件,调用netCDF4库读取此文件绘制气温分布图。下图用月平均气温为例,绘制了全球范围内某月月平均气温图。
1 from mpl_toolkits.basemap import Basemap
2 import matplotlib.pyplot as plt
3 import numpy as np
4 import netCDF4 as nc
5 #调用netCDF4模块读取nc文件
6 ncDATA=nc.Dataset('air.mon.mean.nc')
7 print(ncDATA.variables.keys())
8 lon=ncDATA.variables['lon']
9 lat=ncDATA.variables['lat']
10 tempera=ncDATA.variables['air']
11 lon=np.array(lon)
12 lat=np.array(lat)
13 tempera=np.array(tempera)
14 level_n=5
15 temp=tempera[848][level_n] #根据需要选取需要的时间段以及不同气压下的数据
17 mymap=Basemap()
18 mymap.drawcoastlines()
19 mymap.drawcountries()
21 lon=np.arange(-180,180,2.5)
22 lat=np.arange(90,-92.5,-2.5)
23 Lon,Lat=np.meshgrid(lon,lat)
24 fig=mymap.contourf(Lon,Lat,temp,30,cmap=plt.cm.RdBu_r)
25 mymap.colorbar(fig)
26 plt.show()
问16.1:basemap能不能画全球范围可见卫星数图(用渐变色表示)?
问16.2:怎么画电离层热力图?
答:basemap可以画可见卫星数图和电离层热力图,两者是同类型。以前者为例,需要首先自己划分格网点,然后分别计算各格网点的单天平均可见卫星数,形成格网文件,然后画图。下图以COD20342.sp3文件中给出的GPS、Galileo、GLONASS、BDS和QZSS的卫星轨道参数为例,绘制当天全球范围内单天平均可见卫星数图。为了简化代码量,有的函数只注释了功能,未给出具体代码。
1 #读取SP3文件,把各时刻各卫星位置存储与Satposs中
2 Satposs=readsp3file()
3 Total=[]
4 for lat in range(90,-90,-2):
5 N_sat = []
6 for lon in range(-180, 180, 5):
7 nsat=count_ns_grid(Satposs,lon,lat) #计算各格网点的可见卫星数
8 N_sat.append(nsat)
9 Total.append(N_sat)
11 pllon=np.arange(-180,180,5)
12 pllat=np.arange(90,-90,-2)
13 grid_lon,grid_lat=np.meshgrid(pllon,pllat)
14 mymap = Basemap(lat_0=0, lon_0=0,resolution='i', area_thresh=5000.0)
15 mymap.drawmapboundary(fill_color='skyblue')
16 mymap.drawmeridians(np.arange(0, 360, 60), labels=[1,0,0,1])
17 mymap.drawparallels(np.arange(-90, 90.001, 30), labels=[1,0,0,1])
19 cs=mymap.contourf(grid_lon, grid_lat, Total,cmap=plt.cm.jet)
20 cbar=mymap.colorbar(cs,location='right',pad="5%")
22 plt.show()
问17:能否把测站属性用渐变颜色表述出来?比如希望以DOP值为颜色变量,快速筛选站星分布较好的测站。
答:可以实现。首先根据测站计算当天平均DOP值,然后读取该文件画图。由于全球测站较多,现以北美地区为例,用不同颜色区分DOP大小,方便用户根据DOP值筛选测站。
1 mymap = Basemap(llcrnrlon=-160.,llcrnrlat=0.,urcrnrlon=-60.,urcrnrlat=80., lat_0=40, lon_0=-110,resolution='i', area_thresh=5000.0)
3 mymap.fillcontinents(color='white', lake_color='lightskyblue')
4 mymap.drawmapboundary(fill_color='skyblue')
6 mymap.drawmeridians(np.arange(0, 360, 60), labels=[1,0,0,1])
7 mymap.drawparallels(np.arange(-90, 90.001, 30), labels=[1,0,0,1])
8 #按照DOP值展点
9 x, y = mymap(lons, lats)
10 cm=plt.cm.get_cmap('RdYlBu')
11 sc=plt.scatter(x, y, c=DOP,cmap=cm,zorder=40,s=80)
12 #实现图上标注的微调,防止遮挡
13 for name,lon,lat,dop in zip(sites,x,y,DOP):
14 print (name, lon, lat,dop)
15 if name=='whit' or name =='pie1':
16 plt.text(lon + 1, lat-3, name, rotation=0, fontsize=15)
17 elif name=='cro1' or name =='abmf':
18 plt.text(lon-6, lat-3, name, rotation=0, fontsize=15)
19 else:
20 plt.text(lon+1,lat+1,name,rotation=0,fontsize=15)
22 plt.colorbar(sc)
23 plt.legend(loc=0)
24 picpath = 'sites_DOP.jpg'
25 plt.savefig(picpath, dpi=360)
问18:怎么画卫星星下点轨迹图?
答:可首先根据SP3文件计算当天各时刻各卫星的位置,转换为经纬度后,调用basemap的scatter函数进行绘制。下图将以某一天的SP3文件为例,绘制当天C06,G01和J01三种卫星的星下点轨迹图。由于代码量较多,只给出绘图部分的源码。
1 def draw_ground_track(Satposs):
2 mymap=Basemap()
3 mymap.fillcontinents(color='white', lake_color='lightskyblue')
4 mymap.drawmapboundary(fill_color='skyblue')
5 mymap.drawcoastlines()
6 mymap.drawcountries()
8 latp=[]
9 lonp=[]
10 lat1=[]
11 lon1=[]
12 lat2=[]
13 lon2=[]
14 #所有卫星的所有历元位置存储于Satposs中
15 for fE0 in Satposs:
16 for fb in fE0.inf_fbs:
17 if fb.id == 'C06':
18 latp.append(fb.lat)
19 lonp.append(fb.lon)
20 if fb.id == 'G01':
21 lat1.append(fb.lat)
22 lon1.append(fb.lon)
23 if fb.id == 'J01':
24 lat2.append(fb.lat)
25 lon2.append(fb.lon)
26 #zorder参数可以控制图层上下
27 mymap.scatter(lonp,latp,1,marker='o',color='r',zorder=30)
28 mymap.scatter(lon1,lat1,1,marker='o',color='b',zorder=30)
29 mymap.scatter(lon2,lat2,1,marker='o',color='g',zorder=30)
30 plt.show()
问19:能否绘制卫星分布图(skyplot)?
答:可以。绘制skyplot时不用调用basemap模块。首先计算出各历元各卫星高度角和方位角,然后当卫星可见时,使用极坐标即可绘制卫星分布图(雷达图)。下图给出在2019年1月1日测站abpo上空C13卫星的运行轨迹图。
1 Satposs=readsp3file() #读取当天的SP3文件获取卫星位置
2 #abpo.19o
3 lon=47.22921
4 lat=-19.01830
5 height=1552.9369
6 Total=[]
7 N_sat = []
8 EL=[]
9 AZ=[]
10 ID=[]
11 #各时刻计算的可见卫星数、高度角、方位角以及卫星ID
12 nsat,EL,AZ,ID=count_ns_el_az(Satposs,lon,lat,height)
14 Nepoch=len(EL)
15 Nsat=len(EL[0])
17 ax=plt.subplot(111,projection='polar')
19 for ep in range(1,Nepoch):
20 for sat in range(1,Nsat):
21 if EL[ep-1][sat-1]>0:
22 if ID[sat-1]=='C13':
23 theta=float(AZ[ep-1][sat-1])
24 r=90-float(EL[ep-1][sat-1])
25 c=ax.scatter(theta,r,color='b',marker=".")
26 print('ep:%2d '%ep,ID[sat-1])
28 ax.tick_params('y',labelleft=False)
29 plt.show()
问20:能否像GMT一样在测站分布图外围加黑白框?
答:这是笔者目前用basemap唯一无法达到和GMT一样效果的地方。想要加上可能需要额外写函数实现。
问21.1:背景图能否显示出像谷歌地图或谷歌卫星云图效果?
问21.2:能否在地图上显示出绿化/沙漠信息或者高程信息等三维信息?
答:在basemap模块中,可通过不同函数,实现不同地图背景地图的更换。如,使用arcgisimage可显示类似谷歌卫星云图的效果,使用etopo函数可显示类似浮雕图的高程信息等。下图以类似谷歌地图的显示效果为例给出示例图:
1 from mpl_toolkits.basemap import Basemap
2 import matplotlib.pyplot as plt
4 mymap = Basemap(llcrnrlon=3.75,llcrnrlat=39.75,urcrnrlon=4.35,urcrnrlat=40.15, epsg=5520)
6 mymap.arcgisimage(service='ESRI_Imagery_World_2D', xpixels = 1500, verbose= True)
7 plt.show()
etopo函数绘制由NOAA提供的浮雕地形图。
1 from mpl_toolkits.basemap import Basemap
2 import matplotlib.pyplot as plt
3 import numpy as np
5 plt.figure(figsize=(6, 6))
7 mymap = Basemap(resolution='h', projection='cyl', llcrnrlon=-90, urcrnrlon=-30, llcrnrlat=-60, urcrnrlat=15)
9 mymap.drawmeridians(np.arange(-90, -30+0.001, 30), labels=[0, 0, 0, 1])
10 mymap.drawparallels(np.arange(-60, 15.001, 15), labels=[1, 0, 0, 0])
12 mymap.etopo()
13 #mymap.bluemarble()
14 #mymap.shadedrelief()
15 mymap.drawcoastlines()
17 plt.title('ETOPO')
19 plt.savefig('etopo.png', dpi=360, bbox_inches='tight')
20 plt.show()
bluemarble函数绘制蓝大理石背景地形图。
shadedrelief函数绘制来源于
http://
www-shadedrelief.com
网站的阴影浮雕背景图。
更多用法可参考:
https://
basemaptutorial.readthedocs.io
/en/latest/backgrounds.html#
问22:basemap能否画动图?比如很多测站不同时间的数据,随时间变化闪烁的动图?
答:可以。最简单的方式为每次画之前清除画布,然后将之前的数据叠加之后再一起画出,即可画出动图,下面代码以之前的skyplot为例,以动图的方式画出C13卫星一天的运行轨迹,感兴趣的读者可自行运行。
1 thetas = []
2 rs = []
3 for ep in range(1,Nepoch):
4 for sat in range(1,Nsat):
5 if EL[ep-1][sat-1]>0:
6 if ID[sat-1]=='C13':
7 theta=float(AZ[ep-1][sat-1])
8 r=90-float(EL[ep-1][sat-1])
10 thetas.append(theta)
11 rs.append(r)
12 ax.cla()
13 c=ax.plot(thetas,rs,color='b',marker=".")
14 plt.pause(0.05)
15 print('ep:%2d '%ep,ID[sat-1])
17 ax.tick_params('y',labelleft=False)
6 与其他工具对比
问23:
basemap画图相比Matlab有哪些优点?学习起来哪个更容易?
答:Python和Matlab的语法都很简单易懂,是很容易学习的编程语言。
与M_map相比,basemap和Python的优势在于安装包占用空间小,免费开源。相比动不动占用几十G空间,还需购买正版license的Matlab相比,前者安装下来占用空间不到1G。
问24:
basemap相比于GMT优势在哪?
答:优势在于Python的语法更简单易懂。
画海岸线就是
drawcoastlines()
函数,填充大陆颜色就是
fillcontinents()
函数。很容易理解。
basemap画图时后面的函数功能直接加到图上。GMT4和5版本绘制一张稍复杂的图时,经常需要在原有的代码中增添、删除或修改已有命令的顺序,尤其需要注意
-K
、
-O
以及重定向符号的使用。用户不注意很容易画图失败。这也是basemap较之GMT更方便的地方。
问25:
对那些有固定GMT代码每次只需要修改数据的图(比如测站分布图),basemap有什么优点?
答:basemap在相同场景下同样能固定脚本,只需要修改数据绘制新图。
因为Python语言的简单,basemap的一个优点在于一段时间后再来阅读和修改代码都相当容易,不会出现看不懂的情况,可维护性高。
问26.1
:如果标注测站ID的话,ID会覆盖站点,密集区域会重叠,需要怎么调整?
问26.2
:GMT画全球测站分布图时,为了不让测站名重叠,需要人工微调,basemap有没有更好的解决办法?
答:据笔者所知,为使测站名不重叠,两个工具都需要人工微调测站名的位置,还没有智能到能自己调节的地步。
问27:
basemap主要参考网站和资料有哪些?
答:首先推荐basemap官网:
[1]
https://
basemaptutorial.readthedocs.io
/en/latest/index.html
[2]
https://
matplotlib.org/basemap/
index.html
其次就是本文小结:-)
还有basemap源码。
附:basemap基本函数
basemap 包包含一系列有用的函数,用于绘制物理特征的边界,例如大陆,海洋,湖泊和河流等,以及政治边界,例如国家地区。以下是一些可用绘图功能:
物理边界和水体
drawcoastlines()
:绘制大陆海岸线
drawlsmask()
:绘制高分辨率海陆掩码作为图像,指定土地和海洋颜色。陆地海面掩模源自GSHHS海岸线数据,并且有多个海岸线选项和像素大小可供选择
drawmapboundary()
:绘制地图边界,包括海洋的填充颜色
drawrivers()
:在地图上绘制河流
fillcontinents()
:用给定的颜色填充大陆;可选择用另一种颜色填充湖泊
drawcountries()
:绘制国界
drawstates()
:绘制北美的州界
drawcounties()
:绘制美国县界
drawgreatcircle()
:在两点之间绘制大圆圈
drawparallels()
:绘制指定纬度的线条
drawmeridians()
:绘制指定经度的线条
drawmapscale()
:在地图上绘制比例尺刻度
bluemarble()
:将 NASA 的蓝色大理石图像作为地图背景
shadedrelief()
:将阴影浮雕图像作为地图背景
etopo()
:绘制etopo浮雕图像作为地图背景
warpimage()
:将用户提供的图像投影到地图上
附:basemap更多绘图例子