多边形压缩 Douglas-Peucker算法

多边形压缩Douglas-Peucker算法,上代码!

#include <iostream>
#include <cmath>
#include <utility>
#include <vector>
#include <stdexcept>
#include <opencv2/opencv.hpp> // 包含OpenCV头文件using namespace std;typedef std::pair<double, double> Point;//使用向量发计算高度
double PerpendicularDistance(const Point& pt, const Point& lineStart, const Point& lineEnd)
{double dx = lineEnd.first - lineStart.first;double dy = lineEnd.second - lineStart.second;//Normalizedouble mag = pow(pow(dx, 2.0) + pow(dy, 2.0), 0.5);if (mag > 0.0){dx /= mag; dy /= mag;}double pvx = pt.first - lineStart.first;double pvy = pt.second - lineStart.second;//Get dot product (project pv onto normalized direction)double pvdot = dx * pvx + dy * pvy;//Scale line direction vectordouble dsx = pvdot * dx;double dsy = pvdot * dy;//Subtract this from pvdouble ax = pvx - dsx;double ay = pvy - dsy;return pow(pow(ax, 2.0) + pow(ay, 2.0), 0.5);
}//递归逐一计算
void RamerDouglasPeucker(const vector<Point>& pointList, double epsilon, vector<Point>& out)
{if (pointList.size() < 2)throw invalid_argument("Not enough points to simplify");// Find the point with the maximum distance from line between start and enddouble dmax = 0.0;size_t index = 0;size_t end = pointList.size() - 1;for (size_t i = 1; i < end; i++){double d = PerpendicularDistance(pointList[i], pointList[0], pointList[end]);if (d > dmax){index = i;dmax = d;}}// If max distance is greater than epsilon, recursively simplifyif (dmax > epsilon){// Recursive callvector<Point> recResults1;vector<Point> recResults2;vector<Point> firstLine(pointList.begin(), pointList.begin() + index + 1);vector<Point> lastLine(pointList.begin() + index, pointList.end());RamerDouglasPeucker(firstLine, epsilon, recResults1);RamerDouglasPeucker(lastLine, epsilon, recResults2);// Build the result listout.assign(recResults1.begin(), recResults1.end() - 1);out.insert(out.end(), recResults2.begin(), recResults2.end());if (out.size() < 2)throw runtime_error("Problem assembling output");}else{//Just return start and end pointsout.clear();out.push_back(pointList[0]);out.push_back(pointList[end]);}
}int main()
{vector<Point> pointList;vector<Point> pointListOut;pointList.push_back(Point(0.0, 6.0));pointList.push_back(Point(1.0, 1.1));pointList.push_back(Point(3.0, 5.0));pointList.push_back(Point(4.0, 6.0));pointList.push_back(Point(5.0, 7.0));pointList.push_back(Point(6.0, 8.1));pointList.push_back(Point(7.0, 9.0));pointList.push_back(Point(8.0, 9.0));pointList.push_back(Point(9.0, 9.0));pointList.push_back(Point(2.0, 10.2));cv::Mat canvas(200, 200, CV_8UC3, cv::Scalar(255, 255, 255)); // 创建一个300x300像素的画布// 显示第一个圆for (auto i : pointList) {cv::circle(canvas, cv::Point(i.first*10, i.second*10), 2, cv::Scalar(0, 0, 255), -1);cv::imshow("Canvas", canvas);cv::waitKey(300); // 等待10秒}cv::waitKey(900);RamerDouglasPeucker(pointList, 1.0, pointListOut);cout << "result" << endl;for (size_t i = 0; i < pointListOut.size(); i++){cv::circle(canvas, cv::Point(pointListOut[i].first * 10, pointListOut[i].second * 10), 2, cv::Scalar(255, 0, 0), -1);cv::imshow("Canvas", canvas);cv::waitKey(300); // 等待10秒cout << pointListOut[i].first << "," << pointListOut[i].second << endl;}system("pause");return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/593022.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

c++语句详细介绍

文章目录 前言简单语句语句作用域1. 局部作用域2. 全局作用域3. 命名空间作用域4. 块作用域 条件语句1. If语句2. Else语句3. Else If语句4. Switch语句5. 嵌套条件语句6. 条件运算符 循环语句1. While循环2. Do-While循环3. For循环4. 范围Based For循环&#xff08;C11及以上…

Linux内核--进程管理(七)进程的核心—task_truct

目录 一、引言 二、调度机制介绍 ------>2.1、任务ID ------>2.2、亲缘关系 ------>2.3、任务状态 ------>2.4、任务权限 ------------>2.4.1、capabilities ------>2.5、运行统计 ------------>2.5.1、pidstat ------>2.6、进程调度 -----…

This is probably not a problem with npm.

项目场景&#xff1a; 新创建的vue3项目&#xff0c;根据elementplus官网安装步骤进行按需导入安装&#xff0c;运行项目报错 This is probably not a problem with npm.There is likely additional logging output above. 原因分析&#xff1a; 是elementplus安装版本和自动…

Python随机点名

python随机点名 # 生成 0 ~ 9 之间的随机数 # 导入 random(随机数) 模块 import random print(random.randint(0,9)) 执行以上代码输出结果为: 4 尝试一下 以上实例我们使用了 random 模块的 randint() 函数来生成随机数&#xff0c;你每次执行后都返回不同的数字&a…

什么是Vue.js的响应式系统(reactivity system)?如何实现数据的双向绑定?

Vue.js的响应式系统是指一种能够跟踪数据变化并实时更新相关界面的机制。它是Vue.js框架的核心特性之一。 在Vue.js中&#xff0c;你可以使用数据绑定语法将数据绑定到DOM元素上。当绑定的数据发生变化时&#xff0c;Vue.js会自动监听这些变化并更新相关的DOM元素。 Vue.js实…

玩转贝启科技BQ3588C开源鸿蒙系统开发板 —— 编译构建及此过程中的踩坑填坑(1)

接前一篇文章&#xff1a;玩转贝启科技BQ3588C开源鸿蒙系统开发板 —— 代码下载&#xff08;2&#xff09; 本文主要参考&#xff1a; BQ3588C_代码下载 上一回完成了代码下载&#xff0c;本回开始进行编译构建。 1. 编译构建 &#xff08;1&#xff09;执行prebuilts 在源…

lc338 比特位计数

lc338 比特位计数 问题&#xff1a; 给一个整数n&#xff0c;遍历0-n的每一个值&#xff0c;统计每个值二进制中1的个数&#xff0c;返回长度为n1的数组。 题解&#xff1a; Brian Kernighan’s 算法。这个算法的核心思想是每次去掉二进制中最右边的一个1&#xff0c;直到所有…

JavaScript可选链接

注&#xff1a;本节仍然使用之前的饭店的对象&#xff0c;可以看上几篇文章查看代码 ● 如果我们想要看看饭店周一的开门时间&#xff0c;我们会这么写 console.log(restaurant.openingHours.mon.open);原因是我们在开放时间中并没有定义周一的开放时间&#xff0c;所有会报错…

2024年工作计划与目标怎么写?如何用手机制作工作待办清单

2024年的钟声已经到来&#xff0c;对于上班族来说&#xff0c;制定一份切实可行的工作计划与目标是非常有必要的。但是&#xff0c;很多人不知道2024年工作计划与目标怎么写&#xff1f;其实&#xff0c;关键在于明确目标、细化计划、合理安排时间&#xff0c;以确保每一步都稳…

python协程

背景 很早之前有一篇中断地关于协程的文章&#xff0c;彼时从js的事件循环切入,对比学习python的asyncio,感觉是个很大的知识点&#xff0c;想写的内容很多&#xff0c;遂搁浅。本次&#xff0c;抽空补齐这块的内容。 序 协程是什么? 协程是个非常好的概念&#xff0c;它在用…

BIO和NIO编程(待完善)

目录 IO模型 BIO NIO 常见问题 IO模型 Java共支持3种网络编程IO模式&#xff1a;BIO&#xff0c;NIO&#xff0c;AIO BIO 同步阻塞模型&#xff0c;一个客户端连接对应一个处理线程 代码示例&#xff1a; Server端&#xff1a; public class BioServer {private static …

window下载安装Mongodb数据库

我们先要访问他的官网 https://www.mongodb.com/zh-cn 然后顶部导航栏 选择 (Products/产品) 下的 (Community Edition/社区版) 进入界面后 找到 MongoDB Community Server Download 点击下面的按钮 Select package 然后会弹到这个位置 第一个版本 用系统默认选择的就好 第二…

Redis - 挖矿病毒 db0 库 backup 反复出现解决方案

问题描述 腾讯云的服务器&#xff0c;使用 Docker 部署了 Redis 之后&#xff0c;发现 DB0 中总是出现 4 条 key&#xff0c;分别是 backup01backup02backup03backup04 而自己每次存入 db0 中的数据过一会就会被无缘无故删除掉。 原因分析 挖矿病毒 解决方案 在启动的时候…

MySQL唯一索引失效的注意点

【1.实验环境】 Docker环境下的MySQL MySQL版本: 5.7.37 【2.表结构】 CREATE TABLE t_1 (id int(11) NOT NULL AUTO_INCREMENT,student_id char(120) DEFAULT NULL,course_id char(120) DEFAULT NULL,is_delete int(11) NOT NULL DEFAULT 0,PRIMARY KEY (id),KEY idx_stude…

QuPath学习④ 脚本使用

目录 1&#xff0c;基础学习 脚本打印项目中所有图像 访问当前图像内容 访问图像元数据 访问像素 创建ROI 创建对象&#xff08;使 ROI 可见&#xff09; 多个ROI Working with BufferedImage 使用 ImageJ 命令示例 2 脚本导出数据&#xff08;重点&#xff09; …

React16源码: Hooks源码实现

Hooks 1 &#xff09;概述 Hooks 在 React16.7版本出现的新功能Hooks 改变了整体应用开发的模式&#xff0c;同时开发体验会和以前会变得不一样Hooks 让函数组件具有类组件的能力 在 function component 里面没有this无法保存 state通过 Hooks可以让 function component 代替…

PSoc62™开发板之串口通信

实验目的 1.使用串口和PC机通信&#xff1a;接收和发送信息 2.接收GNSS模块定位信息 实验准备 PSoc62™开发板CH340 USB转TTL模块ATGM332D GNSS模块公母头杜邦线x4 板载资源 板载有多少uart 创建工程例程&#xff0c;在libraries/HAL_Drivers/uart_config.h中查看BSP支持…

跨境电商如何通过API选品文章

跨境电商通过API选品是一个相对复杂的过程&#xff0c;涉及到多个环节和考虑因素。以下是一个简化的概述&#xff0c;介绍了如何通过API进行选品&#xff0c;由于字数限制&#xff0c;这里仅能展示部分内容&#xff0c;如需5000字的文章&#xff0c;可能需要根据这个提纲进一步…

凯越推出复古150踏板欧洲先上?DAE150/150亮相

今天临下班发现凯越在欧洲的官网上更新了一台复古踏板&#xff0c;外观别说还有点精致的意思&#xff0c;一共分为125和150两个配置&#xff0c;都是采用的水冷单缸发动机。 配置和参数等数据简单过一下&#xff0c;这种车型更多的是看外观了&#xff0c;仪表采用的LCD的显示屏…