当前位置: 首页 > article >正文

QGIS插件开发实战:手把手教你用Python调用高德地图API做路径规划(附坐标转换避坑指南)

QGIS插件开发实战Python集成高德地图API的路径规划全流程解析当我们需要在QGIS中实现路径规划功能时高德地图API提供了丰富的服务接口。本文将带你从零开始开发一个能够调用高德地图API进行路径规划的QGIS插件并重点解决开发过程中最棘手的坐标转换问题。1. 开发环境准备与插件框架搭建在开始开发前我们需要配置好开发环境。QGIS插件开发主要依赖以下工具链QGIS 3.x建议使用LTR长期支持版本Python 3.x与QGIS版本匹配的Python环境PyCharm作为主要开发IDEQt Designer用于设计插件界面首先我们需要在PyCharm中配置QGIS的Python解释器。这可以通过创建一个批处理文件来实现echo off SET OSGEO4W_ROOTC:\Program Files\QGIS 3.x call %OSGEO4W_ROOT%\bin\o4w_env.bat call %OSGEO4W_ROOT%\apps\qt5\bin\qt5_env.bat call %OSGEO4W_ROOT%\apps\Python37\Scripts\pyqt5_env.bat path %PATH%;%OSGEO4W_ROOT%\apps\qgis\bin path %PATH%;%OSGEO4W_ROOT%\apps\Python37\Scripts set PYTHONPATH%OSGEO4W_ROOT%\apps\qgis\python;%PYTHONPATH% set QT_PLUGIN_PATH%OSGEO4W_ROOT%\apps\qgis\qtplugins;%QT_PLUGIN_PATH% start PyCharm aware of QGIS /B C:\Program Files\JetBrains\PyCharm\bin\pycharm64.exe %*使用QGIS的Plugin Builder工具可以快速生成插件框架python pb_tool.py create这会生成以下主要文件结构MyPlugin/ ├── __init__.py ├── metadata.txt ├── my_plugin.py ├── my_plugin_dialog.py ├── resources.qrc ├── resources.py └── ui_my_plugin.py2. 高德地图API集成与调用高德地图提供了多种路径规划API接口我们需要先申请开发者密钥。API主要支持以下几种出行方式出行方式API端点必需参数驾车/v3/direction/drivingorigin, destination步行/v3/direction/walkingorigin, destination骑行/v4/direction/bicyclingorigin, destination公交/v3/direction/transit/integratedorigin, destination, cityAPI调用的基本流程如下获取起点和终点的坐标构造API请求URL发送HTTP请求并处理响应解析返回的路径数据以下是驾车路径规划的示例代码def get_driving_route(origin, destination, api_key): url fhttps://restapi.amap.com/v3/direction/driving?origin{origin[0]},{origin[1]}destination{destination[0]},{destination[1]}key{api_key} response requests.get(url) data response.json() if data[status] 1 and data[route][paths]: route data[route][paths][0] distance float(route[distance]) # 单位米 duration float(route[duration]) # 单位秒 steps [] polyline [] for step in route[steps]: step_info { instruction: step[instruction], distance: step[distance], duration: step[duration], road: step.get(road, ) } steps.append(step_info) # 解析polyline坐标 for point in step[polyline].split(;): lon, lat map(float, point.split(,)) polyline.append((lon, lat)) return distance, duration, polyline, steps else: raise Exception(fAPI调用失败: {data.get(info, 未知错误)})3. 坐标转换GCJ-02与WGS84互转高德地图API返回的坐标使用GCJ-02坐标系俗称火星坐标而QGIS通常使用WGS84坐标系。要实现坐标的准确叠加我们需要进行坐标系转换。3.1 WGS84转GCJ-02发送请求前我们需要将QGIS中的WGS84坐标转换为GCJ-02def wgs84_to_gcj02(lon, lat, api_key): url fhttps://restapi.amap.com/v3/assistant/coordinate/convert?locations{lon},{lat}coordsysgpskey{api_key} response requests.get(url) data response.json() if data[status] 1: return map(float, data[locations].split(,)) else: raise Exception(f坐标转换失败: {data.get(info, 未知错误)})3.2 GCJ-02转WGS84高德地图没有提供官方的GCJ-02转WGS84接口我们需要自己实现转换算法。以下是基于迭代法的转换实现import math # 定义常量 a 6378245.0 # 长半轴 f 1 / 298.3 # 扁率 ee 1 - (1 - f) * (1 - f) # 第一偏心率平方 PI math.pi def out_of_china(lon, lat): 判断坐标是否在中国境外 return not (72.004 lon 137.8347 and 0.8293 lat 55.8271) def transform_lat(x, y): 纬度转换 ret -100.0 2.0 * x 3.0 * y 0.2 * y * y 0.1 * x * y 0.2 * math.sqrt(abs(x)) ret (20.0 * math.sin(6.0 * x * PI) 20.0 * math.sin(2.0 * x * PI)) * 2.0 / 3.0 ret (20.0 * math.sin(y * PI) 40.0 * math.sin(y / 3.0 * PI)) * 2.0 / 3.0 ret (160.0 * math.sin(y / 12.0 * PI) 320.0 * math.sin(y * PI / 30.0)) * 2.0 / 3.0 return ret def transform_lon(x, y): 经度转换 ret 300.0 x 2.0 * y 0.1 * x * x 0.1 * x * y 0.1 * math.sqrt(abs(x)) ret (20.0 * math.sin(6.0 * x * PI) 20.0 * math.sin(2.0 * x * PI)) * 2.0 / 3.0 ret (20.0 * math.sin(x * PI) 40.0 * math.sin(x / 3.0 * PI)) * 2.0 / 3.0 ret (150.0 * math.sin(x / 12.0 * PI) 300.0 * math.sin(x * PI / 30.0)) * 2.0 / 3.0 return ret def wgs84_to_gcj02(lon, lat): WGS84转GCJ02 if out_of_china(lon, lat): return lon, lat dLat transform_lat(lon - 105.0, lat - 35.0) dLon transform_lon(lon - 105.0, lat - 35.0) radLat lat / 180.0 * PI magic math.sin(radLat) magic 1 - ee * magic * magic sqrtMagic math.sqrt(magic) dLat (dLat * 180.0) / ((a * (1 - ee)) / (magic * sqrtMagic) * PI) dLon (dLon * 180.0) / (a / sqrtMagic * math.cos(radLat) * PI) gcjLat lat dLat gcjLon lon dLon return gcjLon, gcjLat def gcj02_to_wgs84(lon, lat): GCJ02转WGS84迭代法 g0 (lon, lat) w0 g0 g1 wgs84_to_gcj02(w0[0], w0[1]) w1 (w0[0] - (g1[0] - g0[0]), w0[1] - (g1[1] - g0[1])) delta (w1[0] - w0[0], w1[1] - w0[1]) while abs(delta[0]) 1e-6 or abs(delta[1]) 1e-6: w0 w1 g1 wgs84_to_gcj02(w0[0], w0[1]) w1 (w0[0] - (g1[0] - g0[0]), w0[1] - (g1[1] - g0[1])) delta (w1[0] - w0[0], w1[1] - w0[1]) return w14. 插件功能实现与优化4.1 多线程处理为了避免界面卡顿我们需要使用QThread来实现后台任务处理from PyQt5.QtCore import QThread, pyqtSignal class RoutePlanningWorker(QThread): finished pyqtSignal(list) progress pyqtSignal(int, int) error pyqtSignal(str) log pyqtSignal(str) def __init__(self, od_pairs, mode, api_key, crs_src, parentNone): super().__init__(parent) self.od_pairs od_pairs self.mode mode self.api_key api_key self.crs_src crs_src self.stop_flag False def run(self): try: results [] total len(self.od_pairs) for idx, (origin, destination) in enumerate(self.od_pairs): if self.stop_flag: self.log.emit(用户中断操作) break try: # 坐标转换和路径规划处理 # ... self.progress.emit(idx 1, total) except Exception as e: self.log.emit(f路径规划失败: {str(e)}) continue self.finished.emit(results) except Exception as e: self.error.emit(f任务执行出错: {str(e)})4.2 路径可视化将规划好的路径在QGIS中可视化def create_route_layer(polyline, crs, name路径规划): 创建路径图层 from qgis.core import QgsVectorLayer, QgsFeature, QgsGeometry, QgsLineString # 创建线图层 layer QgsVectorLayer(fLineString?crs{crs.authid()}, name, memory) provider layer.dataProvider() # 创建要素 feature QgsFeature() points [QgsPointXY(lon, lat) for lon, lat in polyline] line QgsLineString(points) feature.setGeometry(QgsGeometry(line)) # 添加要素到图层 provider.addFeature(feature) layer.updateExtents() return layer4.3 结果导出支持将结果导出为CSV文件def export_to_csv(results, file_path): 导出结果到CSV import csv with open(file_path, w, newline, encodingutf-8) as f: writer csv.writer(f) writer.writerow([起点ID, 终点ID, 距离(米), 时间(秒)]) for result in results: writer.writerow([ result[origin_id], result[destination_id], result[distance], result[duration] ])5. 常见问题与调试技巧在开发过程中可能会遇到以下典型问题坐标偏移问题现象路径显示位置与实际位置有偏差解决确保正确执行了GCJ-02和WGS84之间的转换调试方法使用已知坐标点测试转换函数API调用限制高德地图API有调用频率限制解决方案实现请求间隔控制import time for od in od_pairs: # 调用API get_route(od) time.sleep(0.2) # 控制请求频率多线程同步问题现象界面卡顿或崩溃解决确保所有UI操作都在主线程执行使用信号槽机制进行线程间通信内存泄漏长时间运行后QGIS变慢解决及时释放不再使用的图层和对象使用QGIS内置的垃圾回收机制坐标系不一致现象路径显示在错误的位置解决检查所有图层的坐标系设置使用QgsCoordinateTransform进行坐标系转换# 坐标系转换示例 def transform_coordinates(point, src_crs, dest_crs): transform QgsCoordinateTransform(src_crs, dest_crs, QgsProject.instance()) return transform.transform(point)开发QGIS插件集成高德地图API时坐标转换是最关键的技术难点。通过本文介绍的迭代转换算法可以较好地解决GCJ-02与WGS84之间的转换问题。在实际项目中建议对转换结果进行抽样验证确保精度满足需求。

相关文章:

QGIS插件开发实战:手把手教你用Python调用高德地图API做路径规划(附坐标转换避坑指南)

QGIS插件开发实战:Python集成高德地图API的路径规划全流程解析 当我们需要在QGIS中实现路径规划功能时,高德地图API提供了丰富的服务接口。本文将带你从零开始,开发一个能够调用高德地图API进行路径规划的QGIS插件,并重点解决开发…...

从ARMA模型到功率谱估计:一个案例讲透现代信号处理中的‘参数化’与‘非参数化’方法

从振动信号到频谱洞察:ARMA与FFT在工程诊断中的方法论抉择 车间里一台大型离心泵突然发出异常嗡鸣,工程师小王手持采集器记录下这段振动信号。面对屏幕上跳动的波形,他需要回答一个关键问题:这段信号中隐藏的频率特征究竟是什么&a…...

保姆级教程:YOLOv8鹰眼目标检测镜像使用全流程解析

保姆级教程:YOLOv8鹰眼目标检测镜像使用全流程解析 1. 引言:为什么选择YOLOv8鹰眼目标检测? 在智能安防、工业质检、智慧零售等领域,高效准确的目标检测技术正发挥着越来越重要的作用。传统目标检测方案往往面临部署复杂、性能不…...

Get Shit Done:基于上下文工程的AI开发框架解决Claude Code上下文衰退难题

Get Shit Done:基于上下文工程的AI开发框架解决Claude Code上下文衰退难题 【免费下载链接】get-shit-done A light-weight and powerful meta-prompting, context engineering and spec-driven development system for Claude Code by TCHES. 项目地址: https://…...

告别百度云!手把手教你从Keil官网下载安装STM32全系列芯片支持包(附离线包备份技巧)

从Keil官网高效获取STM32芯片支持包的完整指南 为什么需要直接从Keil官网获取芯片支持包 在嵌入式开发领域,STM32系列微控制器因其出色的性能和丰富的生态系统而广受欢迎。然而,许多开发者在搭建开发环境时,往往习惯于从第三方网盘获取Keil M…...

UE5开发者必备:10个免费3D模型资源网站推荐(含避坑指南)

UE5开发者必备:10个免费3D模型资源网站深度评测与实战指南 当你深夜盯着UE5编辑器里那个孤零零的默认立方体发呆时,是否也经历过这种绝望?作为经历过上百个原型项目的老司机,我深刻理解优质3D资源对开发效率的致命影响。市面上90…...

清华团队发布机器人版“GPT时刻”:UniDex让机械手看懂世界,零样本操控万物!

80%成功率,碾压式超越现有方案,灵巧手操控迎来“GPT”时刻这篇论文用一种极其优雅且强大的方式,解决了机器人领域一个长期存在的根本性难题:如何让形态各异、复杂无比的灵巧手,像人类一样,看一眼就能学会使…...

保姆级教程:用VMware和Kali复现Vulnstack红日靶场2的完整渗透流程(附CS联动技巧)

红队实战进阶:Kali与Cobalt Strike协同渗透Vulnstack靶场全解析 环境配置与网络拓扑设计 在开始渗透测试之前,正确的环境搭建是成功的基础。不同于简单的虚拟机启动,专业级红队演练需要精确模拟企业内网环境。我们采用三台靶机(WE…...

突破4大硬件限制:老旧Windows设备升级Windows 11的3维优化方案

突破4大硬件限制:老旧Windows设备升级Windows 11的3维优化方案 【免费下载链接】OpenCore-Legacy-Patcher 体验与之前一样的macOS 项目地址: https://gitcode.com/GitHub_Trending/op/OpenCore-Legacy-Patcher 老旧设备升级Windows 11的价值解析 在数字化快…...

YOLOv8推理慢?CPU深度优化技巧让速度提升2倍

YOLOv8推理慢?CPU深度优化技巧让速度提升2倍 你是不是也遇到过这种情况?部署了YOLOv8模型,功能强大,检测精准,但一到实际推理就卡得不行,CPU占用率飙升,处理一张图片要等好几秒。尤其是在没有G…...

别再硬啃理论了!用STM32F407+OpenMV做个会‘看’会‘动’的小车,代码全开源

从零打造会“思考”的智能小车:STM32F407OpenMV实战指南 当你第一次看到这个小车精准识别路标并自主避障时,那种成就感会瞬间点燃你对嵌入式开发的热情。这不是又一套枯燥的理论教程,而是一个真实可用的智能小车项目——它能用摄像头“看”世…...

毕业不焦虑!百考通AI如何成为你论文季的秘密武器

摘要:面对开题迷茫、逻辑混乱、查重崩溃的经典困局,我如何用百考通AI高效完成了毕业论文的“逆袭”。 深夜三点,室友的鼾声均匀,我屏幕的冷光映照着文档末尾不断闪烁的光标。眼前的文档,除了标题,空空如也。…...

告别“手搓论文”焦虑:百考通AI期刊写作全流程通关秘籍

从选题到投稿,一套工具,帮你避开90%的审稿雷区 在学术研究的漫长旅途中,许多研究者都曾面临这样的困境:精心培育的 idea,扎实的实验数据,却在转化为论文、投向期刊的“最后一公里”屡屡碰壁。不是因为研究本…...

【Cuvil编译器生产级AI推理落地指南】:20年编译器老兵亲授Python模型从PyTorch到裸金属推理的7大避坑红线

第一章:Cuvil编译器在Python AI推理中的应用Cuvil 是一款面向AI工作负载的轻量级领域专用编译器,专为优化Python生态中基于NumPy、Torch和ONNX的推理流程而设计。它不依赖传统JIT或解释器层,而是通过静态图提取、张量算子融合与硬件感知调度&…...

BiliTools全平台高效解决方案:从新手到进阶的B站资源管理指南

BiliTools全平台高效解决方案:从新手到进阶的B站资源管理指南 【免费下载链接】BiliTools A cross-platform bilibili toolbox. 跨平台哔哩哔哩工具箱,支持视频、音乐、番剧、课程下载……持续更新 项目地址: https://gitcode.com/GitHub_Trending/bil…...

除了浏览器点一点,Oracle 19c EM Express还能这么用?几个提升DBA效率的隐藏技巧

Oracle 19c EM Express高阶应用:解锁DBA效率提升的五大隐藏技巧 当你已经能够熟练地通过浏览器访问Oracle 19c EM Express时,是否思考过这个轻量级管理工具还能为你带来哪些惊喜?本文将带你超越基础操作,探索那些被大多数DBA忽略的…...

MT5中文改写工具实操:支持Markdown格式保留与结构化文本增强

MT5中文改写工具实操:支持Markdown格式保留与结构化文本增强 1. 工具概览与核心价值 MT5中文改写工具是一个基于Streamlit和阿里达摩院mT5模型构建的本地化NLP工具。它能对输入的中文文本进行语义改写和数据增强,在保持原意不变的前提下生成多种不同的…...

Fire Dynamics Simulator:火灾动力学模拟的核心引擎与实战应用

Fire Dynamics Simulator:火灾动力学模拟的核心引擎与实战应用 【免费下载链接】fds Fire Dynamics Simulator 项目地址: https://gitcode.com/gh_mirrors/fd/fds 揭示核心价值:为何FDS成为火灾模拟领域的标准工具? 在建筑安全设计、…...

Git-RSCLIP快速部署指南:一键启动,轻松搭建你的遥感图像搜索引擎

Git-RSCLIP快速部署指南:一键启动,轻松搭建你的遥感图像搜索引擎 1. 引言:遥感图像搜索的新利器 在遥感图像分析领域,我们经常面临这样的挑战:如何从海量卫星和航拍图像中快速找到特定目标?传统方法需要人…...

聊天记录丢失?这款开源工具让数据安全不再愁

聊天记录丢失?这款开源工具让数据安全不再愁 【免费下载链接】WeChatExporter 一个可以快速导出、查看你的微信聊天记录的工具 项目地址: https://gitcode.com/gh_mirrors/wec/WeChatExporter 你是否曾经历过手机意外损坏后丢失数年聊天记录的痛苦&#xff1…...

【附源码】FPGA三段式状态机在智能售货系统中的实战解析

1. 智能售货系统与FPGA的完美结合 自动售货机已经渗透到我们生活的各个角落,从地铁站到写字楼,随处可见它们的身影。但你是否想过,这些看似简单的机器背后隐藏着怎样的技术奥秘?作为一名在FPGA领域摸爬滚打多年的工程师&#xff…...

游戏外挂开发实战:用C#和Cheat Engine读取内存实现自动补血(附完整代码)

游戏内存数据解析与自动化操作的技术实现 在数字娱乐领域,游戏开发与逆向工程一直是两个相互促进的技术方向。了解游戏内部数据结构和运行机制,不仅有助于开发者优化产品,也能帮助技术爱好者深入理解计算机系统的工作原理。本文将探讨一种通过…...

精准拓客新范式:号码核验行业的痛点破解与技术赋能之路,氪迹科技法人号码核验系统,阶梯式价格

在B端拓客进入精细化竞争的深水区,“有效线索”成为企业突破增长瓶颈的核心抓手,而号码核验作为拓客流程的前置关键环节,其服务质量直接决定了后续线索转化效率与投入回报比。据Gartner相关报告显示,传统拓客渠道的单个有效线索成…...

告别黑盒:用Python手把手解析SMPP协议PDU,从抓包到解码一条龙

告别黑盒:用Python手把手解析SMPP协议PDU,从抓包到解码一条龙 当你在深夜收到短信网关返回的一串十六进制数据时,是否曾对着Wireshark抓包界面陷入沉思?SMPP协议作为运营商短信系统的"暗语",其二进制PDU结构…...

极速获取全平台歌词:163MusicLyrics跨平台解析工具使用指南

极速获取全平台歌词:163MusicLyrics跨平台解析工具使用指南 【免费下载链接】163MusicLyrics Windows 云音乐歌词获取【网易云、QQ音乐】 项目地址: https://gitcode.com/GitHub_Trending/16/163MusicLyrics 你是否经常遇到想听的歌曲找不到匹配歌词的情况&a…...

革命性AI肖像动画工具LivePortrait:一键让静态照片“动“起来

革命性AI肖像动画工具LivePortrait:一键让静态照片"动"起来 【免费下载链接】LivePortrait Bring portraits to life! 项目地址: https://gitcode.com/GitHub_Trending/li/LivePortrait 你是否曾经想过让老照片中的亲人重新展露笑容?或…...

技术数据解析 | CALCE圆柱电池数据集:SOC估计的OCV测试基准

1. CALCE圆柱电池数据集的核心价值 CALCE电池数据集由马里兰大学先进生命周期工程中心发布,是目前全球最权威的公开电池测试数据之一。这个数据集最吸引我的地方在于它提供了完整的实验环境记录和标准化的测试流程,这对于电池状态估计算法的开发简直是雪…...

【PAT甲级真题】- Shopping in Mars (25)

题目来源 Shopping in Mars (25) 题目描述点击链接自行查看 注意点: 输出时按照区间左端点从小到大输出 思路简介 简单的滑动窗口 我做了一个小处理 因为题目实际上要求找的是大于等于目标值的区间 所以移动左指针的条件写成 l>r&&sum>m 这样我认…...

uni-app实战:驰腾打印机蓝牙对接与二维码打印全解析

1. 为什么选择uni-app对接驰腾打印机? 在移动开发领域,跨平台解决方案越来越受到开发者青睐。uni-app作为一款基于Vue.js的跨平台框架,可以一次开发同时发布到iOS、Android以及各种小程序平台。这种特性使得它成为对接硬件设备的理想选择&am…...

Vitis 2022.1下,Ultrascale+ MPSOC PL端lwIP以太网完整配置流程(含约束文件与时钟设置)

Vitis 2022.1环境下Ultrascale MPSOC PL端lwIP以太网全流程实战指南 当我们需要在Zynq Ultrascale MPSOC平台上实现高性能网络通信时,PL端以太网方案往往能提供比PS端更灵活的设计空间和更高的吞吐量。本文将手把手带你完成从Vivado工程创建到Vitis应用部署的完整流…...