欧拉回路(道路)

16 篇文章 0 订阅

无向欧拉图:

定义:给定无孤立结点图G,若存在一条路,经过图中每边一次且仅一次,该条路称为欧拉路;若存在一条回路,经过图中每边一次且仅一次,该回路称为欧拉回路。具有欧拉回路的图称作欧拉图。

定理:无向图G具有一条欧拉路,当且仅当G是连通的,且有零个或两个奇数度结点。欧拉路和欧拉回路的概念,很易推广到有向图中去。 


有向欧拉图 :
定义:给定有向图G,通过图中每边一次且仅一次的一条单向路(回路),称作单向欧拉路(回路)。


定理:有向图G具有一条单向欧拉回路,当且仅当是连通的,且每个结点入度等于出度。一个有向图G具有单向欧拉路,当且仅当它是连通的,而且除两个结点外,每个结点的入度等于出度,但这两个结点中,一个结点的入度比出度大1,另一个结点的入度比出度小1。


找欧拉路(或回路)可以用Fleury算法,如下:
(1).找一个出发点P (如果图中有两个奇数度的点话,任取其一,有向图取出度大的那一个)。
(2).找一条与P相连的边,伸展欧拉路(选边的时候应注意,最后再选取断边;断边是:当去掉该边后使得图不连通的边)。
(3).将选出的边所连的另一个点取代P,去掉该边,重复(2),直至经过所有的边。

Fleury的另一种说法:

  (1)任取v0∈V(G),令P0=v0;

  (2)设Pi=v0e1v1e2...eivi已经行遍,按下面方法来从E(G)-{e1,e2,...,ei}中选 取 ei+1:

  (a)ei+1与vi想关联;
  (b)除非无别的边可供行遍,否则ei+1不应该为Gi=G-{e1,e2,...,ei}中的桥.
  (3)当(2)不能再进行时,算法停止。

可以证明,当算法停止时所得简单回路Pm = v0e1v1e2...emvm(vm=v0)为G中的一条欧拉回路。


打印出欧拉回路

void euler(int cur) {
    for(int i = 1; i <= nRoads; i++) {    //nRoads代表了路径的条数,枚举所有的路径。G[i][j]:第i个顶点与第j条边相连
        if(!vis[i] && G[cur][i]) {
            vis[i] = true;
            euler(G[cur][i]);
            stack[++top] = i;
        }
    }
}

递归的输出,我觉得得好好的理解一遍,最好是能够模拟一下。



下面的是《算法竞赛入门经典》(刘汝佳)书中的:

注:下面的程序同时适用于欧拉道路和回路。但如果需要打印的是欧拉道路,在主程序调用的时候,参数必须是道路的起点。另外,打印的顺序是逆序的,因此在真正适用这份代码的时候,你应当吧 printf 语句替换成一条 push 语句, 把边(u,v)压入栈中

void euler(int cur) {
    for (int v=0; v<n; v++)
        if (G[u][v] && !vis[u][v]) {
            vis[u][v] = vis[v][u] = 1;
            euler(v);
            printf("%d %d\n", u, v);
        }
}

尽管上面的代码只适用于无向图,但不难改成有向图:把 vis[u][v] = vis[v][u] = 1 改成 vis[u][v] 即可。



更多:http://www.nocow.cn/index.php/%E6%AC%A7%E6%8B%89%E8%B7%AF


  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值