开篇

在做一个Low Poly的课题,而这种低多边形的成像效果在现在设计中越来越被喜欢,其中的低多边形都是由三角形组成的。

而如何自动生成这些看起来很特殊的三角形,就是本章要讨论的内容。

选择

其是最先是由很多离散的点组成,基于这个确定的点集,将点集连接成一定大小的三角形,且分配要相对合理,才能呈现出漂亮的三角化。

这时则要求使用三角剖分算法(Delaunay),引于

【定义】三角剖分:假设V是二维实数域上的有限点集,边e是由点集中的点作为端点构成的封闭线段, E为e的集合。那么该点集V的一个三角剖分T=(V,E)是一个平面图G,该平面图满足条件:

1.除了端点,平面图中的边不包含点集中的任何点。

2.没有相交边。

3.平面图中所有的面都是三角面,且所有三角面的合集是散点集V的凸包。

在实际中运用的最多的三角剖分是Delaunay三角剖分,它是一种特殊的三角剖分。先从Delaunay边说起:

【定义】Delaunay边:假设E中的一条边e(两个端点为a,b),e若满足下列条件,则称之为Delaunay边:存在一个圆经过a,b两点,圆内(注意是圆内,圆上最多三点共圆)不含点集V中任何其他的点,这一特性又称空圆特性。

【定义】Delaunay三角剖分:如果点集V的一个三角剖分T只包含Delaunay边,那么该三角剖分称为Delaunay三角剖分。

【定义】假设T为V的任一三角剖分,则T是V的一个Delaunay三角剖分,当前仅当T中的每个三角形的外接圆的内部不包含V中任何的点。

如图,将离散点联结成Delaunay三角形

算法

关于Delaunay三角形的算法,有翻边算法、逐点插入算法、分割合并算法、Bowyer-Watson算法等。

而在这几种算法中,逐点插入算法比较简单、易懂,在本文中只针对该算法进行讨论,该算法也是目前使用最为广泛的Delaunay算法。

在该算法中,主要应用Delaunay三角形【定义4】,理解下来就是每一个三角形的外接圆圆内不能存在点集内的其它任何一点,而有时候会出现点在外接圆上的情况,这种情况被称为“退化”。

在文章里对该方法进行了分析,并提出了伪代码思路:

subroutine triangulate

input : vertex list

output : triangle list

initialize the triangle list

determine the supertriangle

add supertriangle vertices to the end of the vertex list

add the supertriangle to the triangle list

for each sample point in the vertex list

initialize the edge buffer

for each triangle currently in the triangle list

calculate the triangle circumcircle center and radius

if the point lies in the triangle circumcircle then

add the three triangle edges to the edge buffer

remove the triangle from the triangle list

endif

endfor

delete all doubly specified edges from the edge buffer

this leaves the edges of the enclosing polygon only

add to the triangle list all triangles formed between the point

and the edges of the enclosing polygon

endfor

remove any triangles from the triangle list that use the supertriangle vertices

remove the supertriangle vertices from the vertex list

end

其方法虽然可实现三角化,但是效率还是不太高

在看过

优化后的伪代码为:

input: 顶点列表(vertices)                      //vertices为外部生成的随机或乱序顶点列表

output:已确定的三角形列表(triangles)

初始化顶点列表

创建索引列表(indices= new Array(vertices.length))    //indices数组中的值为0,1,2,3,......,vertices.length-1

基于vertices中的顶点x坐标对indices进行sort         //sort后的indices值顺序为顶点坐标x从小到大排序(也可对y坐标,本例中针对x坐标)

确定超级三角形

将超级三角形保存至未确定三角形列表(temp triangles)

将超级三角形push到triangles列表

遍历基于indices顺序的vertices中每一个点//基于indices后,则顶点则是由x从小到大出现

初始化边缓存数组(edge buffer)

遍历temp triangles中的每一个三角形

计算该三角形的圆心和半径

如果该点在外接圆的右侧

则该三角形为Delaunay三角形,保存到triangles

并在temp里去除掉

跳过

如果该点在外接圆外(即也不是外接圆右侧)

则该三角形为不确定//后面会在问题中讨论

跳过

如果该点在外接圆内

则该三角形不为Delaunay三角形

将三边保存至edge buffer

在temp中去除掉该三角形

对edge buffer进行去重

将edge buffer中的边与当前的点进行组合成若干三角形并保存至temp triangles中

将triangles与temp triangles进行合并

除去与超级三角形有关的三角形

end

大多数同学看过伪代码后还是一头雾水,所以用图来解释这个过程,我们先用三点来做实例:

如图,随机的三个点

根据离散点的最大分布来求得随机一个超级三角形(超级三角形意味着该三角形包含了点集中所有的点)

我的方法是根据相似三角形定理求得与矩形一半的小矩形的对角三角形,扩大一倍后则扩大后的直角三角形斜边经过点(Xmax,Ymin)

但是为了将所有的点包含在超级三角形内,在右下角对该三角形的顶点进行了横和高的扩展,并要保证这个扩展三角形底大于高,才能实现包含

这样求得的超级三角形不会特别大使得计算复杂,而且过程也简单,并将超级三角形放入temp triangles中

接下来就像是伪代码中描述的那样,对temp triangle中的的三角形遍历画外接圆,这时先对左边的第一个点进行判断,其在圆内

所以该三角形不为Delaunay三角形,将其三边保存至edge buffer中,temp triangle中删除该三角形

将该点与edge buffer中的每一个边相连,组成三个三角形,加入到temp triangles中

再将重复对temp triangles的遍历并画外接圆,这时使用的是第二个点来进行判断

该点在三角形1外接圆右侧,则表示左侧三角形为Delaunay三角形,将该三角形保存至triangles中

该点在三角形2外接圆外侧,为不确定三角形,所以跳过(后面会讲到为什么要跳过该三角形),但并不在temp triangles中删除

该点在三角形3外接圆内侧,则这时向清空后的edge buffer加入该三角形的三条边,并用该点写edge buffer中的三角边进行组合,组合成了三个三角形并加入到temp triangles中

再次对temp triangles进行遍历,这里该数组里则含有四个三角形,一个是上次检查跳过的含有第一个点的三角形和新根据第二个点生成的三个三角形

该点在三角形1外接圆右侧,则该三角形为Delaunay三角形,保存至triangles中,并在temp triangles中删除

该点在三角形2外接圆外侧,跳过

该点在三角形3外接圆内侧,将该三边保存至temp buffer中,并在temp triangles中删除

该点在三角形4外接圆内侧,将该三边保存至temp buffer中,并在temp triangles中删除

这时,temp buffer 中有六条边,triangles中有两个三角形,temp triangles中有1个三角形

对temp buffer中的六条边进行去重,得到五条边,将该点与这五条边组合成五个三角形并加入到temp triagnles 中,这时temp triangles中有6个三角形

由于三个点已经遍历结束,到了不会再对第三个点形成的三角形做外接圆,这时则将triangles与temp trianlges合并,合并后的数组表示包含已经确定的Delaunay三角形和剩下的三角形

这时除去合并后数组中的和超级三角形三个点有关的所有三角形,即进行数组坐标的限定,则得到了最后的结果:

这是用最少的三个点来做讲解,点数越多的话计算量会越大,但是都是在上面步骤下进行的。

问题

在用点对三角形外接圆位置关系进行判断的时候,为什么点在外接圆的右侧的话可以确定该三角形是Delaunay三角形

而当点外接圆的外侧且非右侧时,为什么要路过三角形,不把该三角形确定为Delaunay三角形呢?

首先,我们在开始的时候对原始方法进行优化时,我们增加了一个indices数组来操作vertices,并对indices依据vertices的x坐标进行了从小到大的排序

则我们在后面遍历点时是从点集的最左侧开始的,如图:

当遍历下一个点时,该点在外接圆的右侧,则表示以后所有的点都在该外接圆的右侧,则保证了Delaunay三角形的空圆特性

而当点在外接圆外,并非外接圆右侧时,如图:

在该三角形的外切圆中,当遍历到点1时,符合在外侧的条件,但是不能确定后面所有的点都保持在外接圆外侧

如果说该三角形就为Delaunay三角形的话,如图中的点2及后面可能出现的点很有可能出现在圆内,而使该三角形被按边分解

在我们的算法中,如果碰到在点在外侧且非右侧的话,会跳过,该三角形一直在temp triangles中被检验,直到碰到下一个点在圆内或圆右才会从temp triangles中去除,进行后面的操作

而当点在圆上时,也是根据在圆内的方法对其进行操作,实际情况中会出现这种情况,上文也讲过,称为“退化”。

最后,附一张delaunay的随机demo图:

The end.

python三角网格代码_三角剖分算法(delaunay)相关推荐

  1. python三角网格代码_Python 实现 Delaunay Triangulation

    Delaunay Triangulation 是一种空间划分的方法,它能使得分割形成的三角形最小的角尽可能的大,关于 Delaunay Triangulation 的详细介绍,请参考这里,Delaun ...

  2. python计算圆周率代码_用算法实现计算圆周率-几何概型估算圆周率Python编程小程序...

    我在http://tieba.baidu.com/p/5953188922?traceid=看见了一个用 编程计算圆周率的方法 具体详细可以打开链接 发现原作者:百度用户阿泰 C531T因为编程语言的 ...

  3. python雪花下落代码_雪花算法python实现

    雪花算法-Snowflake Snowflake是Twitter提出来的一个算法,其目的是生成一个64bit的整数: 1bit:一般是符号位,不做处理 41bit:用来记录时间戳,这里可以记录69年, ...

  4. python三角网格代码_python中shapely的多多边形三角网格/网格

    我想你应该用三角形来填充这个区域.你的"三角形"只是其中的一半.如果只需要你的那一半,只需注释掉for循环的第二部分.在import numpy as np from shapel ...

  5. python双目视觉三维重建代码_双目立体视觉的三维重建方法与流程

    本发明为双目立体视觉的三维重建方法,属于图像处理与机器视觉领域,涉及halcon软件,具体涉及空间场景的三维重建. 背景技术: 三维重建技术作为计算机视觉领域的一个重要分支,在人工智能.虚拟现实.非接 ...

  6. 神经网络变学习率算法代码_机器学习算法之神经网络

    点击蓝字关注我哦 人工神经网络(Artificial Neural Networks,ANN)系统是 20 世纪 40 年代后出现的.它是由众多的神经元可调的连接权值连接而成,具有大规模并行处理.分布 ...

  7. 层次聚类 matlab代码_聚类算法解析一

    01 概述 本次针对聚类算法进行讲解,因为内容较多,会分多篇文章进行讲解,主要的内容包括聚类算法的整体介绍,针对不同类别的聚类算法比如划分聚类.层次聚类.密度聚类等算法进行介绍,在讲解每类算法时会结合 ...

  8. python新年有趣代码_搞几款由“Python”语言编写的“有趣、恶搞、好玩”的程序代码!...

    下载好向圈APP可以快速联系圈友 您需要 登录 才可以下载或查看,没有帐号?立即注册 x 为提高大家对"Python"编程语言的学习兴趣,今天给大家分享几款有趣的Python程序代 ...

  9. python游走代码_介绍一个全局最优化的方法:随机游走算法(Random Walk)

    1. 关于全局最优化求解 全局最优化是一个非常复杂的问题,目前还没有一个通用的办法可以对任意复杂函数求解全局最优值.上一篇文章讲解了一个求解局部极小值的方法--梯度下降法.这种方法对于求解精度不高的情 ...

  10. k邻近算法python代码_机器学习算法之K近邻法-Python实现

    一.算法简介 k近邻法(k-nearest neighbor,k-NN)是一种基本的分类方法,输入的是实例的特征向量,对应于特征空间的点,输出结果为实例的类别,可以取多类.对于训练集来说,每个实例的类 ...

最新文章

  1. SharePoint 2010中的客户端AJAX应用——ASP.NET AJAX模板
  2. 动态slimmable网络:高性能的网络轻量化方法!对比slimmable涨点5.9%
  3. flask读取数据库(mysql)并展示表格(讲解获取表头的方法)【附上flask好看点的helloworld】
  4. linux权限管理(chown、chgrp、chomd)
  5. [转]九个Console命令,让js调试更简单
  6. Escape The Maze (easy version) 多源最短路,bfs(1700)
  7. 语言怎么绘画人物肖像_坦培拉绘画技法——油画简史
  8. 如何显示Organization unit ID
  9. 164. Maximum Gap
  10. 2021 年前端学习路线总结
  11. IDEA配置好maven后新建maven项目一直build失败的解决方法
  12. JavaScript:typeof的用法以及undefined和null的区别
  13. 阶段3 2.Spring_04.Spring的常用注解_2 常用IOC注解按照作用分类
  14. 信安小组 第三周 总结
  15. 前东家欠款 1.4 亿?我看贝店爆雷事件
  16. mysql误删除数据恢复处理
  17. 一大波苹果CMS系统主题来袭
  18. python麦克劳林级数展开
  19. 腾创秒会达分布式无线全向麦克风MHD-G3B-13M分体式大功率扬声器
  20. java.lang.IllegalStateException: Ambiguous handler methods

热门文章

  1. Goole 和 Bing 和 Baidu 搜索语法
  2. coreldraw常用快捷键
  3. 算法训练 - 黑色星期五 有些西方人比较迷信,如果某个月的13号正好是星期五,他们就会觉得不太吉利,用古人的说法,就是“诸事不宜”。请你编写一个程序,统计出在某个特定的年份中,出现了多少次既是13号又
  4. 性能测试---影响性能的因素
  5. Ques1017勇者斗恶龙
  6. Python.PyAutoGUI.利用手机模拟器破解Yousee密码
  7. 微信\支付宝扫码条码区分规则
  8. 使用canvas压缩图片大小
  9. windows驱动 - IRQL
  10. 定性特征转化为定量特征之factorize