利用Python的folium包绘制城市道路图

2023-12-01 14:10

本文主要是介绍利用Python的folium包绘制城市道路图,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

写在前面

很长一段时间内,我都在研究在线地图的开发者文档,百度地图和高德地图的开发者中心提供了丰富的在线地图服务,虽然有一定的权限限制,但不得不说,还是给我的科研工作提供了特别方便的工具,在博客前面我先放上这两个在线地图开放平台的web API的地址链接:

  • 百度地图开放平台
  • 高德地图开放平台

基于这两个平台,博主进行了一系列的开发研究工作,本文介绍其中一项技术,如何用folium包绘制城市道路图,当然,也可绘制非城市道路图,只要提供正确的路名就行了。
开发工具:

  • Python3.7
  • Spyder编译器(也可以用pycharm,不过建议用Spyder,因为编译过程中产生的变量太多,基本上都是json数据,我都是一边看一边写,这里Spyder优势明显)
  • chrome浏览器

folium介绍及相关设置

folium基础功能

folium的开发包在这里
简单来说,它是一个地理信息可视化的包,目前除了pyecharts,我用的最多的就是这个包,支持在在线地图上添加点、线、面等要素,而且还支持画热力图,不过热力图的效果真心不咋地,我看中的是它添加点、线、面形状的功能,而且各种要素可以设置颜色、大小、文字标记等属性,可视化效果还是不错的。这篇博客,也是应用了它画线的功能,绘制道路轮廓线。
在这里插入图片描述
上图图片来源:https://www.jianshu.com/p/32ec6afcc7a6?utm_campaign=hugo
比较遗憾的是,目前folium支持的地图底图有限,像openstreetmap是支持的,高德也支持,但不支持百度地图。当然,不支持百度地图并不能限制咱们开发者的脚步,本博客中也有相关介绍。
pip install folium
使用之前,先安装一下这个包。

folium参数设置

先看两行代码:

import folium
line_road = folium.Map(location=[31.596730,120.233516],zoom_start=15,tiles = 'http://webrd02.is.autonavi.com/appmaptile?lang=zh_cn&size=1&scale=1&style=7&x={x}&y={y}&z={z}',attr = 'default')
  • location参数,设置展示地图的中心坐标点,就是说,比如你想看无锡市,可以设置成无锡市市中心的经纬度坐标
  • zoom_start是地图缩放等级,最高差不多可以到19还是20,如果想看大场景,就设小一点,想看局部地图就设大一点
  • tiles这个参数很重要,设置的是你的地图格式,默认的是OpenStreetMap,我这里把它改成了’http://webrd02.is.autonavi.com/appmaptile?lang=zh_cn&size=1&scale=1&style=7&x={x}&y={y}&z={z}’,表示我用的是高德地图作为底图。为啥要改呢,因为每个不同的地图公司,用的坐标系不一样,高德地图和google地图、soso地图、aliyun地图、mapabc地图所用坐标相同,都是国测局(GCJ02)坐标,和百度地图用的坐标系不一样,如果直接拿百度坐标系下的经纬度画在高德地图上,那就会整体偏移,使用之前必须进行坐标转换
  • 其他用默认参数

获取道路参数

本博客的目的是画道路的轮廓图,首先必须得有数据才能画图。博主知道,目前这些地图公司,都是自己把车在开在路上去采集路上的经纬度,只要我获取到了这些经纬度,那我不就能绘制道路了吗?本着这个想法,我就到处搜索资料,开发者的潜能是无限了,同样在CSDN上我找到了一篇博客,链接在这里 ,真的很棒,不过他是用js写的,无所谓,方法是通的,用这篇博文提供的接口,真的实现了在地图上绘制道路的功能。
但是,但是,但是,,,
用了一段时间后,这个功能被封了,为此,我特意联系了高德地图开发者中心,他们的解释如下:
在这里插入图片描述

意思就是,这个功能用不了了,花钱也别想用。
其实很正常,这个功能太牛逼了。
虽然用不了了,但我还是介绍一下怎么实现的,万一以后又能用了呢。

高德地图获取道路经纬度的API介绍

先看接口:

http://restapi.amap.com/v3/road/roadname?parameters

这个接口和高德地图其他功能的接口一样,后面的parameters是需要写的参数,每个参数之间用&隔开,其中keywords是道路名,这个参数必填,当然还有key也是必填的。现在来看看这一段的具体怎么写,比如我想获取的是无锡市钱荣路的经纬度:

# -*- coding: utf-8 -*-
"""
Created on Mon Mar 30 16:54:32 2020@author: HP
"""import json
import pandas as pd
from urllib.request import urlopen, quote
import folium
import numpy as nproad = quote('钱荣路')
key = YourKey  # 换成你自己申请的key
url = 'http://restapi.amap.com/v3/road/roadname?city=0510&key=%s&keywords=%s' % (key, road)req = urlopen(url)
res = req.read().decode()
temp = json.loads(res)roads = temp['roads']
pos = []
# 由于道路可能分段,比如钱荣路会分成钱荣路普通段和钱荣路高架,这都属于钱荣路的路段,因此必须要都取出来
for p in range(len(roads)):pos.extend(roads[p]['polylines'])pos_cal = []
line_qrroad = folium.Map(location=[31.596730,120.233516],zoom_start=15,tiles = 'http://webrd02.is.autonavi.com/appmaptile?lang=zh_cn&size=1&scale=1&style=7&x={x}&y={y}&z={z}',attr = 'default')for i in range(len(pos)):m = pos[i].split(';')lat_lon = []for j in range(len(m)):n = m[j].split(',')n = list(map(float, n))n[0],n[1] = n[1],n[0]lat_lon.append(n)pos_cal.append(n)folium.PolyLine(lat_lon,weight = 5, color = 'red',opacity = 0.8).add_to(line_qrroad)
line_qrroad.save('lineqrroad.html')    map_qrroad = folium.Map(location=[31.596730,120.233516],zoom_start=15,tiles = 'http://webrd02.is.autonavi.com/appmaptile?lang=zh_cn&size=1&scale=1&style=7&x={x}&y={y}&z={z}',attr = 'default')for point in range(len(pos_cal)):folium.CircleMarker(location=[pos_cal[point][0],pos_cal[point][1]],radius=4,popup='popup',color='red',fill=True,fill_color='red').add_to(map_qrroad)map_qrroad.save('render.html')

现在这段程序,已经没法解析出经纬度了,运行的话,会报如下错误:

runfile('D:/python/folium/qianrongroad.py', wdir='D:/python/folium')
Traceback (most recent call last):File "D:\python\folium\qianrongroad.py", line 26, in <module>roads = temp['roads']KeyError: 'roads'

意思就是说,没有‘road’这个key,我试图把请求串输入浏览器,返回的结果如下:

{"info":"INSUFFICIENT_PRIVILEGES","infocode":"10012","status":"0","sec_code_debug":"d41d8cd98f00b204e9800998ecf8427e","key":"ea12ed719e4ed13862dd0876384c6512","sec_code":"d41d8cd98f00b204e9800998ecf8427e"}

说我没有足够的权限。
好了,暂且不说了,看看代码的意思
前面是常规的json数据解析,没啥好说的,只要接口正常,就能取出数据来。

# 由于道路可能分段,比如钱荣路会分成钱荣路普通段和钱荣路高架,这都属于钱荣路的路段,因此必须要都取出来
for p in range(len(roads)):pos.extend(roads[p]['polylines'])

上面这个循环,注释已经解释清楚了,一条路可能会被高德分成好几部分,当然这是科学的,比如完整的钱荣路是分成了钱荣路普通路段和高架路段的。也就是说解析出来的roads的长度是2,分别是roads[0]和roads[1],而经纬度数据则在roads[p][‘polylines’]里面。

for i in range(len(pos)):m = pos[i].split(';')lat_lon = []for j in range(len(m)):n = m[j].split(',')n = list(map(float, n))n[0],n[1] = n[1],n[0]lat_lon.append(n)pos_cal.append(n)

这里是数据的分析,看起来写的很简单,其实很复杂,可惜没有数据来配套解释了。首先,经度和纬度之间用的是’,‘分割,每一小段路之间用的是’;‘来分割,这个’;'分割我理解为高德对数据的一种加密方式,完整的一条路被高德划分成了很多小段,我把数据取出来后,自己用matplotlib演示了一下完整的路的绘制过程,看下面几张动图(没法插视频):
在这里插入图片描述
在这里插入图片描述
为什么我这里要这么做,因为只有这样,我才是真正的理解了这些解析出来的数据是怎么连成一条完整的道路的,这样才好到folium中去绘制道路,实际上就是循环绘制,每一小段一小段的画,最后会连成一条完整的道路,过程就是下面这段代码:

for point in range(len(pos_cal)):folium.CircleMarker(location=[pos_cal[point][0],pos_cal[point][1]],radius=4,popup='popup',color='red',fill=True,fill_color='red').add_to(map_qrroad)

循环可以简化,博主习惯了写range(len)这种方式
渲染成网页,就可以打开了,看下结果:
在这里插入图片描述
忽略图中的圆圈标记,是我添加的其他信息。
放大看细节:
在这里插入图片描述
很良心有木有,双向车道、辅道、支路全部都有了,可惜当初没有把数据保存下来,只保存了这么个图。
这样就完事儿了。
自从高德把这个接口封了之后,博主神伤了好久,想了各种办法,连付费使用都想出来了,但是高德一个字,不给用、没权限、有钱也不行。没办法,项目还要继续,功能还要继续实现。想到之前百度地图事业部某年轻有为的负责人来咱们单位交流过,一番交涉,发现百度地图API也没有公开这个功能,但是离线地图可以。于是,,,博主又开始忙活了。
因涉及相关隐私,博主不具体介绍。总之,一番操作,获取到了百度地图坐标系下的道路经纬度数据,但前面说了,folium不支持百度地图,强行用百度地图坐标系下的经纬度坐标数据是会出乱子的,但这点小问题难不倒博主,高德地图API有坐标转换的接口呢。

百度坐标系下的坐标点转换成高德坐标系下的坐标点

接口在这里
接着上代码

import json
from urllib.request import urlopen, quote
import folium
import osdef BaiduMap2AMap(data):polylines = []for i in range(len(data)):poly = []for j in range(len(data[i])):url = 'https://restapi.amap.com/v3/assistant'\'/coordinate/convert?locations=%f,%f&coordsys=baidu'\'&key=YourKey'%(data[i][j][1], data[i][j][0])req = urlopen(url)res = req.read().decode()temp = json.loads(res)location = temp['locations'].split(',')location = list(map(float,location))location[0], location[1] = location[1], location[0] poly.append(location)polylines.append(poly)return polylines

常规的接口访问和数据分析代码,不做过多解释,不过给大家看一下函数的输入格式:
在这里插入图片描述
结合数据格式,大家应该能看明白这段代码
用相同的方法来画地图,看看结果
在这里插入图片描述
不错哦,再看看细节:
在这里插入图片描述
细节不如之前丰富,不过也很不错了。
再给大家看看,如果直接用百度坐标系下的经纬度点画到高德地图上是个啥效果:
在这里插入图片描述
看到没,整体偏了不少,所以坐标转换很重要。。

结语

写这篇博客,也是为了记录最近一段时间的工作。这篇博客实际上没解决任何问题,根本没有取到任何有用的数据。但是,其中坐标转换那个函数还是有点用的。里面的数据分析方法,也是博主花了一些心血才写出来的。
不断学习,不断进步吧。

这篇关于利用Python的folium包绘制城市道路图的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/441468

相关文章

Python脚本实现自动删除C盘临时文件夹

《Python脚本实现自动删除C盘临时文件夹》在日常使用电脑的过程中,临时文件夹往往会积累大量的无用数据,占用宝贵的磁盘空间,下面我们就来看看Python如何通过脚本实现自动删除C盘临时文件夹吧... 目录一、准备工作二、python脚本编写三、脚本解析四、运行脚本五、案例演示六、注意事项七、总结在日常使用

Python将大量遥感数据的值缩放指定倍数的方法(推荐)

《Python将大量遥感数据的值缩放指定倍数的方法(推荐)》本文介绍基于Python中的gdal模块,批量读取大量多波段遥感影像文件,分别对各波段数据加以数值处理,并将所得处理后数据保存为新的遥感影像... 本文介绍基于python中的gdal模块,批量读取大量多波段遥感影像文件,分别对各波段数据加以数值处

python管理工具之conda安装部署及使用详解

《python管理工具之conda安装部署及使用详解》这篇文章详细介绍了如何安装和使用conda来管理Python环境,它涵盖了从安装部署、镜像源配置到具体的conda使用方法,包括创建、激活、安装包... 目录pytpshheraerUhon管理工具:conda部署+使用一、安装部署1、 下载2、 安装3

Python进阶之Excel基本操作介绍

《Python进阶之Excel基本操作介绍》在现实中,很多工作都需要与数据打交道,Excel作为常用的数据处理工具,一直备受人们的青睐,本文主要为大家介绍了一些Python中Excel的基本操作,希望... 目录概述写入使用 xlwt使用 XlsxWriter读取修改概述在现实中,很多工作都需要与数据打交

使用Python实现在Word中添加或删除超链接

《使用Python实现在Word中添加或删除超链接》在Word文档中,超链接是一种将文本或图像连接到其他文档、网页或同一文档中不同部分的功能,本文将为大家介绍一下Python如何实现在Word中添加或... 在Word文档中,超链接是一种将文本或图像连接到其他文档、网页或同一文档中不同部分的功能。通过添加超

Python MySQL如何通过Binlog获取变更记录恢复数据

《PythonMySQL如何通过Binlog获取变更记录恢复数据》本文介绍了如何使用Python和pymysqlreplication库通过MySQL的二进制日志(Binlog)获取数据库的变更记录... 目录python mysql通过Binlog获取变更记录恢复数据1.安装pymysqlreplicat

利用Python编写一个简单的聊天机器人

《利用Python编写一个简单的聊天机器人》这篇文章主要为大家详细介绍了如何利用Python编写一个简单的聊天机器人,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 使用 python 编写一个简单的聊天机器人可以从最基础的逻辑开始,然后逐步加入更复杂的功能。这里我们将先实现一个简单的

基于Python开发电脑定时关机工具

《基于Python开发电脑定时关机工具》这篇文章主要为大家详细介绍了如何基于Python开发一个电脑定时关机工具,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 简介2. 运行效果3. 相关源码1. 简介这个程序就像一个“忠实的管家”,帮你按时关掉电脑,而且全程不需要你多做

Python实现高效地读写大型文件

《Python实现高效地读写大型文件》Python如何读写的是大型文件,有没有什么方法来提高效率呢,这篇文章就来和大家聊聊如何在Python中高效地读写大型文件,需要的可以了解下... 目录一、逐行读取大型文件二、分块读取大型文件三、使用 mmap 模块进行内存映射文件操作(适用于大文件)四、使用 pand

python实现pdf转word和excel的示例代码

《python实现pdf转word和excel的示例代码》本文主要介绍了python实现pdf转word和excel的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价... 目录一、引言二、python编程1,PDF转Word2,PDF转Excel三、前端页面效果展示总结一