js代码实现地图轨迹点抽稀 Douglas-Peuker(道格拉斯-普克)抽稀算法

本文主要是介绍js代码实现地图轨迹点抽稀 Douglas-Peuker(道格拉斯-普克)抽稀算法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

缘起:

目前在开发展示船舶轨迹的应用,有很多地图展示页面,其中一项就是播放轨迹。然而轨迹点太多了,七天的数据就有一万八千多个,点过多会影响性能,于是想到了抽稀。

算法:

目前来说轨迹抽稀较为常用的算法有:步长法线段过滤法Douglas-Peuker算法以及垂距限值法。Douglas-Peuker相对来说精度不错,国内大部分开发者也都在用,所以就采用了这个算法。大概搜了下,有很多语言的实现,唯独没找到js代码实现的,所以就自己写了一个。

Douglas-Peuker算法:

  1. 在曲线首尾两点A,B之间连接一条直线AB,该直线为曲线的弦;
  2. 得到曲线上离该直线段距离最大的点C,计算其与AB的距离d;
  3. 比较该距离与预先给定的阈值threshold的大小,如果小于threshold,则该直线段作为曲线的近似,该段曲线处理完毕。
  4. 如果距离大于阈值,则用C将曲线分为两段AC和BC,并分别对两段取信进行1~3的处理。
  5. 当所有曲线都处理完毕时,依次连接各个分割点形成的折线,即可以作为曲线的近似。

代码实现:

计算两点之间的距离

calculationDistance: function (point1, point2) {let lat1 = point1.geometry.y;let lat2 = point2.geometry.y;let lng1 = point1.geometry.x;let lng2 = point2.geometry.x;let radLat1 = lat1 * Math.PI / 180.0;let radLat2 = lat2 * Math.PI / 180.0;let a = radLat1 - radLat2;let b = (lng1 * Math.PI / 180.0) - (lng2 * Math.PI / 180.0);let s = 2 * Math.asin(Math.sqrt(Math.pow(Math.sin(a / 2), 2)+ Math.cos(radLat1) * Math.cos(radLat2) * Math.pow(Math.sin(b / 2), 2)));return s * 6370996.81;}

 计算点pX到点pA和pB所确定的直线的距离

distToSegment: function (start, end, center) {let a = Math.abs(this.calculationDistance(start, end));let b = Math.abs(this.calculationDistance(start, center));let c = Math.abs(this.calculationDistance(end, center));let p = (a + b + c) / 2.0;let s = Math.sqrt(Math.abs(p * (p - a) * (p - b) * (p - c)));return s * 2.0 / a;}

递归方式压缩轨迹

compressLine: function (coordinate, result, start, end, dMax) {if (start < end) {let maxDist = 0;let currentIndex = 0;let startPoint = coordinate[start];let endPoint = coordinate[end];for (let i = start + 1; i < end; i++) {let currentDist = this.distToSegment(startPoint, endPoint, coordinate[i]);if (currentDist > maxDist) {maxDist = currentDist;currentIndex = i;}}if (maxDist >= dMax) {//将当前点加入到过滤数组中result.push(coordinate[currentIndex]);//将原来的线段以当前点为中心拆成两段,分别进行递归处理this.compressLine(coordinate, result, start, currentIndex, dMax);this.compressLine(coordinate, result, currentIndex, end, dMax);}}return result;}

供调用的抽稀入口函数

douglasPeucker: function (coordinate, dMax) {//轨迹点抽希if (!coordinate || !(coordinate.length > 2)) {return [];}//coordinate.forEach((item, index) => {//    item.id = index;//});var result = gisCommon.compressLine(coordinate, [], 0, coordinate.length - 1, dMax);result.push(coordinate[0]);result.push(coordinate[coordinate.length - 1]);var resultLatLng = result.sort(gisCommon._compare("utc"));//resultLatLng.forEach((item) => {//    item.id = undefined;//});return resultLatLng;}

 排序方法

_compare: function (prop) {return function (obj1, obj2) {	                 var val1 = obj1.attributes[prop];var val2 = obj2.attributes[prop];if (!isNaN(val1) && !isNaN(val2)) {val1 = Number(val1);val2 = Number(val2);}if (val1 < val2) {return -1;} else if (val1 > val2) {return 1;} else {return 0;}};}

上面的代码为了排序添加了id,最后去掉了,万一你的点对象里面有‘id’这个属性不想被覆盖,改个名字就行

转 作者:逆水行舟丶
来源:简书

这篇关于js代码实现地图轨迹点抽稀 Douglas-Peuker(道格拉斯-普克)抽稀算法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java实现检查多个时间段是否有重合

《Java实现检查多个时间段是否有重合》这篇文章主要为大家详细介绍了如何使用Java实现检查多个时间段是否有重合,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录流程概述步骤详解China编程步骤1:定义时间段类步骤2:添加时间段步骤3:检查时间段是否有重合步骤4:输出结果示例代码结语作

使用C++实现链表元素的反转

《使用C++实现链表元素的反转》反转链表是链表操作中一个经典的问题,也是面试中常见的考题,本文将从思路到实现一步步地讲解如何实现链表的反转,帮助初学者理解这一操作,我们将使用C++代码演示具体实现,同... 目录问题定义思路分析代码实现带头节点的链表代码讲解其他实现方式时间和空间复杂度分析总结问题定义给定

Java覆盖第三方jar包中的某一个类的实现方法

《Java覆盖第三方jar包中的某一个类的实现方法》在我们日常的开发中,经常需要使用第三方的jar包,有时候我们会发现第三方的jar包中的某一个类有问题,或者我们需要定制化修改其中的逻辑,那么应该如何... 目录一、需求描述二、示例描述三、操作步骤四、验证结果五、实现原理一、需求描述需求描述如下:需要在

如何使用Java实现请求deepseek

《如何使用Java实现请求deepseek》这篇文章主要为大家详细介绍了如何使用Java实现请求deepseek功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1.deepseek的api创建2.Java实现请求deepseek2.1 pom文件2.2 json转化文件2.2

Java调用DeepSeek API的最佳实践及详细代码示例

《Java调用DeepSeekAPI的最佳实践及详细代码示例》:本文主要介绍如何使用Java调用DeepSeekAPI,包括获取API密钥、添加HTTP客户端依赖、创建HTTP请求、处理响应、... 目录1. 获取API密钥2. 添加HTTP客户端依赖3. 创建HTTP请求4. 处理响应5. 错误处理6.

python使用fastapi实现多语言国际化的操作指南

《python使用fastapi实现多语言国际化的操作指南》本文介绍了使用Python和FastAPI实现多语言国际化的操作指南,包括多语言架构技术栈、翻译管理、前端本地化、语言切换机制以及常见陷阱和... 目录多语言国际化实现指南项目多语言架构技术栈目录结构翻译工作流1. 翻译数据存储2. 翻译生成脚本

如何通过Python实现一个消息队列

《如何通过Python实现一个消息队列》这篇文章主要为大家详细介绍了如何通过Python实现一个简单的消息队列,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录如何通过 python 实现消息队列如何把 http 请求放在队列中执行1. 使用 queue.Queue 和 reque

Python如何实现PDF隐私信息检测

《Python如何实现PDF隐私信息检测》随着越来越多的个人信息以电子形式存储和传输,确保这些信息的安全至关重要,本文将介绍如何使用Python检测PDF文件中的隐私信息,需要的可以参考下... 目录项目背景技术栈代码解析功能说明运行结php果在当今,数据隐私保护变得尤为重要。随着越来越多的个人信息以电子形

使用 sql-research-assistant进行 SQL 数据库研究的实战指南(代码实现演示)

《使用sql-research-assistant进行SQL数据库研究的实战指南(代码实现演示)》本文介绍了sql-research-assistant工具,该工具基于LangChain框架,集... 目录技术背景介绍核心原理解析代码实现演示安装和配置项目集成LangSmith 配置(可选)启动服务应用场景

使用Python快速实现链接转word文档

《使用Python快速实现链接转word文档》这篇文章主要为大家详细介绍了如何使用Python快速实现链接转word文档功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 演示代码展示from newspaper import Articlefrom docx import