多边形填充-活动边表法

news/2024/10/6 20:12:53/文章来源:https://www.cnblogs.com/yimeimanong/p/18289424

参考文档:
参考1:https://blog.csdn.net/u013044116/article/details/49737585
参考2:https://blog.csdn.net/keneyr/article/details/83747501

算法思想:
对多边形沿y轴从0开始遍历,建立边表NET。只记录顶点的x, dx, ymax。
根据NET构建活动边表AET(activate edge table)。射线与多边形的交点坐标为(x + △x, y + 1), △x是斜率的倒数(dx)。详解见 参考1。

实现代码:

点击查看代码
# encoding=utf8
import mathimport cv2
import numpy as npclass EdgeTable:def __init__(self, x, dx, ymax):self.x = xself.dx = dxself.ymax = ymaxself.next = Nonedef fill_polys(polys):poly_x_max = float('-inf')poly_y_max = float('-inf')for poly in polys:poly_x_max = max(poly_x_max, poly[0])poly_y_max = max(poly_y_max, poly[1])poly_x_max += 1poly_y_max += 1point_count = len(polys)# build edge tablenet = [None for _ in range(poly_y_max)]for i in range(poly_y_max):for j in range(point_count):# 一个点跟前面的点形成线段 跟后边的点也形成线段if polys[j][1] == i:if polys[(j - 1 + point_count) % point_count][1] > polys[j][1]:x_ = polys[j][0]dx_ = (polys[(j - 1 + point_count) % point_count][0] - polys[j][0]) / (polys[(j - 1 + point_count) % point_count][1] - polys[j][1])ymax_ = polys[(j - 1 + point_count) % point_count][1]pnt = EdgeTable(x_, dx_, ymax_)pnt.next = net[i]net[i] = pntif polys[(j + 1 + point_count) % point_count][1] > polys[j][1]:x_ = polys[j][0]dx_ = (polys[(j + 1 + point_count) % point_count][0] - polys[j][0]) / (polys[(j + 1 + point_count) % point_count][1] - polys[j][1])ymax_ = polys[(j + 1 + point_count) % point_count][1]pnt = EdgeTable(x_, dx_, ymax_)pnt.next = net[i]net[i] = pnt# build activate edge tablebg = np.zeros((poly_x_max, poly_y_max), dtype=np.uint8)aet = EdgeTable(0, 0, 0)for i in range(poly_y_max):# 计算新的交点 更新aet表pa = aet.nextwhile pa:pa.x = pa.x + pa.dxpa = pa.next# 更新aet表之后对ate表进行排序# 断表排序 不开辟新空间pa = aetpcur = aet.nextpa.next = Nonewhile pcur:while pa.next and pcur.x >= pa.next.x:pa = pa.nextpt = pcur.nextpcur.next = pa.nextpa.next = pcurpcur = ptpa = aet# net表中的点加入到aet表中,并用插值法按x排序插入pn = net[i]pa = aetwhile pn:while pa.next and pn.x >= pa.next.x:pa = pa.nextpt = pn.nextpn.next = pa.nextpa.next = pnpn = ptpa = aet# 填充pcur = aet.nextwhile pcur and pcur.next:# 浮点数转成整数 如果dx是负数 向下取整;如果dx是正数 向上取整s = math.ceil(pcur.x) if pcur.dx > 0 else math.floor(pcur.x)e = math.ceil(pcur.next.x) if pcur.next.dx > 0 else math.floor(pcur.next.x)# for x in range(pcur.x, pcur.next.x+1):for x in range(s, e + 1):bg[x][i] = 1pcur = pcur.next# 从aet表中删除ymax==i的结点pa = aetpcur = pa.nextwhile pcur:if pcur.ymax == i:pa.next = pcur.nextdel pcurpcur = pa.nextelse:pcur = pcur.nextpa = pa.nextprint(bg)def matrix(polys):m = np.zeros((10, 10), dtype=np.uint8)for x, y in polys:m[x, y] = 1return mdef cv_fill(polys):mask = np.zeros((5, 5), dtype=np.uint8)cv2.fillPoly(mask, [np.array(polys)], 1, lineType=cv2.FILLED)return maskif __name__ == "__main__":# r = matrix([[8, 8], [2, 7], [5, 5], [2, 2], [5, 1], [9, 3]])# print(r)polys = [[2, 4], [1, 2], [3, 0], [4, 1], [3, 2], [4, 3]]fill_polys(polys)# cv_fill(polys)

结果展示:
image

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

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

相关文章

WebOffice在线编微软Offfice,并以二进制流的形式打开Word文档

在日常办公场景中,我们经常会遇到这种场景:我们的合同管理系统的各种Word,excel,ppt数据都是以二进制数组的形式存储在数据库中,如何从数据库中读取二进制数据,以二进制数据作为参数,然后加载到浏览器的Office窗口,实现在线编辑Office的功能呢? 猿大师办公助手是猿大师旗…

香橙派编译linux内核支持ebpf和虚拟WIFI

前言 上一篇文章香橙派5plus上跑云手机方案一 redroid(带硬件加速)中说了怎么运行redroid,这篇补一下怎么修改参数编译内核。 补充 上篇文章有个内容需要补充一下:更新完内核需要用下面的命令防止内核被apt更新,不然后面使用apt update又回到官方的内核(注意版本号,当前是1…

太强了!这款仿微信的聊天工具开源了!

盒子IM —— 一个仿微信实现的网页版聊天软件,支持私聊、群聊、离线消息、发送语音、图片、文件、emoji 表情等功能,不依赖任何第三方收费组件。大家好,我是 Java陈序员。 问君能有几多愁,开源项目解千愁! 今天,给大家介绍一个开源的仿微信聊天工具,同时支持 PC 端、移动…

Python 进度条tqdm

在编写Python脚本时,尤其是那些需要处理大量数据或者执行耗时操作的脚本,监控任务进度是非常重要。 tqdm的安装pip install tqdm -i https://pypi.tuna.tsinghua.edu.cn/simple/基本用法from tqdm import tqdm import timefor i in tqdm(range(100)):time.sleep(0.1) # 模拟…

zotero驯服第一期

初始学习网址:https://blog.csdn.net/m0_56896669/article/details/136580119 遇到问题:脚注里面有引用还有编号,PDF引用格式错误 Zotero中文社区:https://zotero-chinese.com/user-guide/ --相当于这个软件的百科全书 But 找了半天没找到想要的,用不明白,算了,…

一个难忘的json反序列化问题

前言 最近我在做知识星球中的商品秒杀系统,昨天遇到了一个诡异的json反序列化问题,感觉挺有意思的,现在拿出来跟大家一起分享一下,希望对你会有所帮助。 案发现场 我最近在做知识星球中的商品秒杀系统,写了一个filter,获取用户请求的header中获取JWT的token信息。 然后根…

windows mysql执行sql文件

背景 快速导入数据表或者数据库。 解决 直接执行sql文件。虽然直接复制内容也行,但是还是执行文件更好一些。登录mysql -u root -p -D xxx-D指定数据库的名称。如果不写,可以在进入mysql命令行后,使用use xxx来使用数据库。执行source xxx.sql特别注意,哪怕路径里有空格,外…

GERL论文阅读笔记

Graph Enhanced Representation Learning for News Recommendation论文阅读笔记 这篇文章是2020年的,也算是比较老的了,但是比较经典,这里来读一下 Abstract 存在的问题: ​ 现有的新闻推荐方法通过从新闻内容和用户与新闻的直接交互(如点击)中建立精确的新闻表征来实现个…

根轨迹不受增益的影响

%fileName:figure10_57.m s=tf(s); K=[25 125 250 500]; sysD=500*(s+4); rlocus(sysD); grid on sysGq=0.4*(s+0.25)/((s-1.6+2.8*j)*(s-1.6-2.8*j)*(s+3.4)*(s+20)); rlocus(sysGq); grid on %sysGz=feedback(series(sysD,sysGq),1); %应该是开环传递函数的根轨迹,这个闭环的…

Spring Boot3整合Mybatis Plus,数据库为MySQL

项目结构如下:注意不需要任何XML文件 1.导入依赖 除了Spring Boot创建时自带的依赖,还需要加入: <!-- Mybatis Plus依赖--> <dependency> <groupId>com.baomidou</groupId> <artifactId>mybatis-plus-spring-boot3-starter</artifactId>…

将.nuget文件夹从C盘移到其它盘,再也不用担心的C盘爆红了

将.nuget文件夹从C盘移到其它盘,再也不用担心的C盘爆红了 命令 查看缓存文件夹的具体路径 dotnet nuget locals all --list第一步 在文件资源管理器输入%AppData%\NuGet\NuGet.Config ,默认存放NuGet.Config的路径,具体情况可能根据你的电脑来操作。 增加配置<config>…

网络测速

网络测速🥏 https://speed.cloudflare.com/