欧拉路和欧拉圈,简言之就是,从无向图的一个结点出发,走一条路/圈,每条边恰好经过一次,即一笔画问题

欧拉定理:一个无向图最多只有两个奇结点,那么我们就从一个奇结点出发,到另一个结点为之,一定有一条欧拉路。

无向图:最多只能有两个奇节点的图则可判定为有欧拉路

有向图:最多只能有两个结点的入度和出度不相同,必须其中一个结点的入度比出度大1(终点),另一个结点的出度比入度大1(起点),且其无向图(即底图)是连通的。

应用:

  1. 判定欧拉路/圈的有无:根据底图的连通性+结点的度数判定,其中连通性的判定可使用DFS(见种子填充)或者并查集
  2. 如果有,构造欧拉路/圈:使用下面的代码:
 //用DFS构造欧拉路/圈
void euler(int u){
for(int v=;v<n;v++) if(G[u][v]&&!vis[u][v]){
vis[u][v]=vis[v][u]=;
euler(v);
printf("打印边:%d %d",u,v);
}
}

UVa中有一个类似的欧拉路的题目,涉及了复杂图的欧拉路的判定(有复边)

大意如下:给定N个单词(1<=N<=100000)判断能否构成词语接龙,即后一个单词的首字母与前一个单词的首字母一致。一个单词每出现一次就要考虑一次。详细的题目描述见传送门

传送门:UVa 10129

思路如下:

  1. 根据要求,我们要判定给定的N个单词能否构成词语接龙,那么我们只需要关注每个单词的开头字母和结尾字母。因此我们可以将每个单词看作从首字母到尾字母的一条边,那么词语接龙就是要在所给的有向图中找到一条欧拉路或者一个欧拉圈。
  2. 判定有向图的条件见上方:1.底图连通。2.节点的度数满足上述条件。
  3. 用并查集来判定底图的连通性(dfs真是烦),p数组用于存储并查集的代表元
  4. p数组存储的是边的节点的代表元,当所有节点的代表元都一致,说明底图连通
  5. 入度和出度分别使用in和out两个数组进行记录,用G作为记录边是否出现的map,每条边只计算一次即可,减少并查集的判别量。
  6. 除了判定欧拉路,也可能存在欧拉圈,因此代码中也判定了欧拉圈

代码如下:

 #include<cstdio>
#include<cstring> //memset()的定义
#include<iostream>
using namespace std; #define MAX 26 int G[MAX][MAX],m;//G存储图,m存储边的数量,用于并查集
int u[MAX*MAX],v[MAX*MAX],p[MAX]; //边i 的端点为u,v,p[x]是用于并查集的标记
int in[MAX],out[MAX]; //记录每个端点的入度和出度
int t,n;//题目中的t和n int char2int(char c){
return c-'a';
}
//Union Find Set
int find(int x){
return p[x]==x?x:p[x]=find(p[x]);
}
//判断以单词为图的连通性
bool isConnect(){
for(int i=;i<MAX;i++)p[i]=i;//初始化并查集
for(int i=;i<m;i++){ //按照边将端点的代表元进行合并
int x=find(u[i]),y=find(v[i]);
if(x!=y)p[x]=y;//Union 这样的Union使得连通性的判断实际上就是无向图的连通性的判断
}
int flag=find(u[]);//将第一个边的u端点作为代表检测元
for(int i=;i<m;i++)if(find(u[i])!=flag) return false;//如果出现了不同的代表元,则说明图不连通
return true;
}
bool hasStartEnd(){
int start=-,end=-;
for(int i=;i<MAX;i++){
if(end< && in[i]==out[i]+)end=i;
else if(start< && in[i]+==out[i])start=i;
else if(in[i]!=out[i])return false;//其他点如果入度和出度不相同,则不可能有欧拉路或者欧拉圈
}
return start==-&&end==- || start>=&&end>=;//两者同时为-1时是欧拉圈,同时大于等于0是欧拉路
}
bool hasEuler(){
if(n==)return true; //如果只有一个单词,那么肯定是可以得
return isConnect()&&hasStartEnd();//注意顺序不能反,从定义上不能反
}
int main(){
string s;
scanf("%d",&t);
while(t--){
scanf("%d",&n);
memset(G,,sizeof(G));m=;//初始化边的数目和图
memset(in,,sizeof(in));//初始化入度出度数组
memset(out,,sizeof(out));
while(n--){
cin>>s;
int u1=char2int(s[]),v1=char2int(s[s.length()-]);//顶点u和v
if(!G[u1][v1]){
u[m]=u1,v[m]=v1;//将边m的端点记录下来,便于使用并查集 ,也相当于初始化了边的端点集合
G[u1][v1]=;m++;//做判断是为了并查集的边数尽可能的小,所以用了if
}
in[v1]++,out[u1]++;
}
hasEuler()?puts("Ordering is possible."):puts("The door cannot be opened.");
}
}

 今天就到这里啦~欧拉路和欧拉圈就搞定啦~see you~(●'◡'●)

欧拉图和欧拉圈-Play On Words(UVa10129)的更多相关文章

  1. 欧拉图 欧拉回路 欧拉通路 Euler

    欧拉图 本文链接:http://www.cnblogs.com/Ash-ly/p/5397702.html 定义: 欧拉回路:图G的一个回路,如果恰通过图G的每一条边,则该回路称为欧拉回路,具有欧拉回 ...

  2. 欧拉图 欧拉回路 欧拉通路 Euler的认识 (转)

    转:https://www.cnblogs.com/Ash-ly/p/5397702.html 定义: 欧拉回路:图G的一个回路,如果恰通过图G的每一条边,则该回路称为欧拉回路,具有欧拉回路的图称为欧 ...

  3. SGU101 求有重边的无向图欧拉迹

    题意:好多木棒,俩端有数字(0--6)标记,按数字相同的端首尾相连成一条直线(木棒可以相同).即求有重边的无向图欧拉迹. 先判定是否为欧拉图,俩个条件,不说了.如果是欧拉图,输出路经. 方法:dfs遍 ...

  4. nyoj 42 一笔画 欧拉通路

    http://acm.nyist.net/JudgeOnline/problem.php?pid=42 一笔画问题 时间限制:3000 ms  |  内存限制:65535 KB 难度:4 描述 zyc ...

  5. The Best Path HDU - 5883 欧拉通路

    图(无向图或有向图)中恰好通过所有边一次且经过所有顶点的的通路成为欧拉通路,图中恰好通过所有边一次且经过所有顶点的回路称为欧拉回路,具有欧拉回路的图称为欧拉图,具有欧拉通路而无欧拉回路的图称为半欧拉图 ...

  6. hdu2588 GCD (欧拉函数)

    GCD 题意:输入N,M(2<=N<=1000000000, 1<=M<=N), 设1<=X<=N,求使gcd(X,N)>=M的X的个数.  (文末有题) 知 ...

  7. BZOJ 2705: [SDOI2012]Longge的问题 [欧拉函数]

    2705: [SDOI2012]Longge的问题 Time Limit: 3 Sec  Memory Limit: 128 MBSubmit: 2553  Solved: 1565[Submit][ ...

  8. BZOJ 2818: Gcd [欧拉函数 质数 线性筛]【学习笔记】

    2818: Gcd Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 4436  Solved: 1957[Submit][Status][Discuss ...

  9. Euler-Maruyama discretization("欧拉-丸山"数值解法)

    欧拉法的来源 在数学和计算机科学中,欧拉方法(Euler method)命名自它的发明者莱昂哈德·欧拉,是一种一阶数值方法,用以对给定初值的常微分方程(即初值问题)求解.它是一种解决常微分方程数值积分 ...

随机推荐

  1. P1338 末日的传说

    题目描述 只要是参加jsoi活动的同学一定都听说过Hanoi塔的传说:三根柱子上的金片每天被移动一次,当所有的金片都被移完之后,世界末日也就随之降临了. 在古老东方的幻想乡,人们都采用一种奇特的方式记 ...

  2. final变量属性小记

    final 修饰符对于类成员变量来说,具备语法上不可变的特性:对于类成员方法来说,具备语法上子类不可覆盖重写的特性(能被继承的前提下). 但 final 并不限制子类对父类被修饰声明的成员变量进行覆盖 ...

  3. 如何把阿里云的服务器配置为mac的共享文件夹(亲测有效)

    写在开头的就是,我只能百分之九十确定这个是真的有效....毕竟试了太多的方法,最后莫名其妙的就好了.. - -# 基础的步骤就不说了,网上一搜一大把,大家可能follow了所有的步骤以后发现还是连接不 ...

  4. 于是他错误的点名开始了 [Trie]

    于是他错误的点名开始了 题目背景 XS中学化学竞赛组教练是一个酷爱炉石的人. 他会一边搓炉石一边点名以至于有一天他连续点到了某个同学两次,然后正好被路过的校长发现了然后就是一顿欧拉欧拉欧拉(详情请见已 ...

  5. 7月24号day16总结

    一开始显示出现问题,js路径不能应用,因为用的是springMVC框架书写,所以有路径的保护和静态引用地址时需要注意的地方 今天进行了最后项目的优化,包括map清洗数据部分的代码和echarts显示的 ...

  6. 函数实现多个按钮控制一个div

    <!DOCTYPE HTML><html><head> <meta http-equiv="txttent-Type" txttent=& ...

  7. ActiveMQ(2) ActiveMQ创建HelloWorld

    启动ActiveMQ: 请参见:ActiveMQ(1) 初识ActiveMQ 创建Maven工程: pom文件: <project xmlns="http://maven.apache ...

  8. [05]Git查看、删除、重命名远程分支和tag

    Git查看.删除.重命名远程分支和tag 2015-06-15:加入姊妹篇: 2013-11-06:加入重命名远程分支的内容: 2013-01-09:加入删除远程tag的内容. 姊妹篇:使用Git.G ...

  9. 在Ubuntu下安装IntelliJ IDEA

    1)在IDEA官网下载Linux版本安装包 2)将安装包shell到/usr/local目录下 3)切到安装目录下,验证文件校验和,官网上显示的校验和: 3d77ee82094dab51e345f16 ...

  10. Quartus ModelSim联合仿真中的RAM初始化

    Modelsim只支持Hex格式的初始化文件,文件需要放在仿真的根目录下,例如:.\simulation\modelsim:并且在利用Quartus宏生成IP时,选择的初始化文件必须用绝对路径!否则M ...