找出平面上的特殊无向图中的所有三角形的算法

找出平面上的特殊无向图中的所有三角形的算法


问题提出背景:在非结构化三角形网格生成过程中,若采用前沿推进法,在推进过程中是不好构造三角形的(而且也没有要),最好在把所有的边都连好以后再找出所有三角形,于是提出了问题:在由三角形构成的平面无向图中如何找出所有三角形?

    网格如图:



要注意的是,这个无向图很特殊,
1.这个图在平面上。
2.这个图是由三角形构成的(如果不是由三角行构成,那这个网格就没有用处了)。

我的算法如下,伪代码表示:
foreach(点 p in所有的点){
foreach(点 np in p的所有邻居点){
foreach(点 nnpin np的所有邻居点){
if(       p,np,nnp三点两两不重合 
&& p,np,nnp三点两两相连 
&& p==uniqPointOfTriangle(p,np,nnp) 
&& uniqPointOf2Points(np,nnp)==np)  ){
输出p,np,nnp构成的三角形。
}
 
}
}

}
 

算法的关键在于uniqPointOfTriangle( )和uniqPointOf2Points( )这两个函数。

这两个函数的原理相同, uniqPointOfTriangle( )uniqPointOf2Points()唯一的作用是

它的一个性质:    输出和输入参数的顺序无关

如果没有这两个函数的判断,每个三角形会被输出6次,而有了这两个函数的限制后,强制在3个元素的6中排列中指定1种,

就消除了重复。

uniqPointOfTriangle的实现我想了一个邪恶的办法,吐舌头

struct point * uniqPointOfTriangle(struct point * a,struct point * b ,struct point * c){
        /*返回一个3个点的特征点,即不管a,b,c的次序,这个函数返回结果唯一*/
        struct point * p=NULL;

        //我直接比较指针大小,O(∩_∩)O哈哈~
        p=(a>b?a:b);
        return p>c?p:c;
}
struct point * uniqPointOfTriangle(struct point * a,struct point * b ,struct point * c){ /*返回一个3个点的特征点,即不管a,b,c的次序,这个函数返回结果唯一*/ struct point * p=NULL; //我直接比较指针大小,O(∩_∩)O哈哈~ p=(a>b?a:b); return p>c?p:c; }
还有一种正常一点的办法:

 

思路是:对三个点,先在x方向找出最小的点,若有一个,直接返回;若有两个,找出y方向小的那个返回。


另外,这样输出的三角形中其内部可能有其他的点,若要消除,再加上一层过滤,去除掉那些"p有邻点在p,np,nnp三角形中的"情况即可,

这是因为这个图由三角形构成的特殊性质,如果有在p--np--nnp中有点,假设这些点都不和p相连,那么,

这些点和p--np, p--nnp构成的区域必然不是三角形!所以可以这样干。

转载于:https://www.cnblogs.com/windydays/archive/2011/08/22/2298106.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要使用Floyd算法找到无向图的最小环,可以先使用Floyd算法求出图中任意两个顶点之间的最短路径,然后再遍历所有的三角形(三个顶点组成的环)找到最小的环。 以下是一个使用Python实现Floyd算法求解无向图最小环的示例: ```python INF = float('inf') def floyd(graph): n = len(graph) dist = [[INF] * n for _ in range(n)] # 初始化距离矩阵 next_vertex = [[-1] * n for _ in range(n)] # 记录最短路径上的下一个顶点 # 初始化距离矩阵和下一个顶点矩阵 for i in range(n): for j in range(n): if i == j: dist[i][j] = 0 elif graph[i][j] != 0: dist[i][j] = graph[i][j] next_vertex[i][j] = j # 使用Floyd算法计算最短路径和下一个顶点 for k in range(n): for i in range(n): for j in range(n): if dist[i][j] > dist[i][k] + dist[k][j]: dist[i][j] = dist[i][k] + dist[k][j] next_vertex[i][j] = next_vertex[i][k] return dist, next_vertex def find_min_cycle(graph): n = len(graph) min_cycle_length = INF dist, next_vertex = floyd(graph) # 遍历所有三角形,找到最小的环 for u in range(n): for v in range(u+1, n): for w in range(v+1, n): if graph[u][v] != 0 and graph[v][w] != 0 and graph[w][u] != 0: cycle_length = dist[u][v] + dist[v][w] + dist[w][u] - 2 * (graph[u][v] + graph[v][w] + graph[w][u]) if cycle_length < min_cycle_length: min_cycle_length = cycle_length return min_cycle_length ``` 使用上述代码,可以通过传入一个邻接矩阵表示的无向图,来计算出最小环的长度。其,`graph`是一个二维列表,表示图的邻接矩阵,如果两个顶点之间没有边,则对应位置的值为0。 注意,这里的最小环指的是由三个顶点组成的环,并且每个顶点都与相邻的两个顶点相连。如果图中不存在这样的环,则返回无穷大(INF)。 希望能够帮助到你!如有更多问题,请继续提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值