题目

输入n(n≤100000)个单词,是否可以把所有这些单词排成一个序列,使得每个单词的第一个字母和上一个单词的最后一个字母相同(例如 acm,malform,mouse)。每个单词最多包含1000个小写字母。输入中可以有重复单词。

解题思路

把字母看作结点,单词看作有向边,则问题有解等价于图中存在欧拉道路。有向图中存在欧拉道路的条件有两个:一是底图(忽略边的方向后得到的无向图)连通,二是度数满足不存在奇点或奇点数为2。度数判读只要在输入时记录每个顶点的入度出度,而连通性判断有两种:DFS和并查集。

代码实现

dfs判断连通性+特判入出度

 #include<stdio.h>
#include<cstring>
using namespace std; const int maxn = + ;
int G[maxn][maxn],in[maxn],out[maxn];
int vis[maxn]; //点是否访问,不是边
int n;
char word[ + ]; void dfs(int u)
{
vis[u] = ;
for (int v = ; v < maxn; v++) if (G[u][v])
{
G[u][v] = G[v][u] = ;
//G[u][v]--; G[v][u]--;
dfs(v);
}
} int main()
{
int T;
scanf("%d", &T);
while (T--)
{
memset(G, , sizeof(G));
memset(in, , sizeof(in));
memset(out, , sizeof(out));
memset(vis, , sizeof(vis));
scanf("%d", &n);
int start; //起点
while (n--)
{
scanf("%s", word);
int len = strlen(word);
int u = word[] - 'a', v = word[len - ] - 'a';
start = u;
vis[u] = vis[v] = -; //出现过的标为-1
G[u][v] = G[v][u] = ; //连通性按无向图处理
//G[u][v]++; G[v][u]++;
out[u]++; //度数按有向图处理
in[v]++;
} bool flag = true; //满足要求为true
int s_odd = ,t_odd = ; //起始奇点、结束奇点
for (int i = ; i < maxn; i++)
{
if (in[i] == out[i]) continue;
if (out[i] == in[i] + && !s_odd) { start = i; s_odd = ; }
else if (in[i] == out[i] + && !t_odd) t_odd = ;
else { flag = false; break; }
}
if (flag)
{
dfs(start); //也可以不从奇点出发,这个只是判断连通性
for (int i = ; i < maxn; i++)
if (vis[i] == -) { flag = false; break; }
} if (flag) printf("Ordering is possible.\n");
else printf("The door cannot be opened.\n");
}
return ;
}

并查集判断连通性+特判入出度

 #include<stdio.h>
#include<iostream>
#include<algorithm>
#include<cstring>
#include<string>
#include<cmath>
using namespace std; const int maxn = + ;
int in[maxn], out[maxn], flag[maxn], p[maxn], fa[maxn];
int n; void init()
{
for (int i = ; i < ; i++)
fa[i] = i;
memset(in, , sizeof(in));
memset(out, , sizeof(out));
memset(flag, , sizeof(flag));
memset(p, , sizeof(p));
}
int find(int x)
{
if (fa[x] != x) return fa[x] = find(fa[x]);
return fa[x];
} void unite(int x, int y)
{
int rx = find(x);
int ry = find(y);
fa[rx] = ry;
} int main()
{
int T;
int a, b;
string str;
scanf("%d", &T);
while (T--)
{
init();
scanf("%d", &n);
for (int i = ; i < n; i++)
{
cin >> str;
a = str[] - 'a';
b = str[str.size() - ] - 'a';
unite(a, b);
in[a]++;
out[b]++;
flag[a] = flag[b] = ;
} int cnt = ; //记录连通分量
int root;
for (int i = ; i < ; i++)
{
if (flag[i])
{
root = find(i);
break;
}
}
for (int i = ; i < ; i++)
{
if (flag[i])
if (root != find(i)) cnt = ;
} if (cnt) {
printf("The door cannot be opened.\n");
continue;
} int k = ; //p[i]记录度数不等的
for (int i = ; i < ; i++)
{
if (flag[i] && in[i] != out[i]) p[k++] = i;
}
if (k == )
{
printf("Ordering is possible.\n");
continue;
}
if (k == && (in[p[]] - out[p[]] == && in[p[]] - out[p[]] == -) || (in[p[]] - out[p[]] == - && in[p[]] - out[p[]] == ))
{
printf("Ordering is possible.\n");
continue;
}
else
{
printf("The door cannot be opened.\n");
}
}
return ;
}

参考链接:https://blog.csdn.net/qq_29169749/article/details/51111377

UVA10129———欧拉道路的更多相关文章

  1. UVA-10129 Play on Words (判断欧拉道路的存在性)

    题目大意:给出一系列单词,当某个单词的首字母和前一个单词的尾字母相同,则这两个单词能链接起来.给出一系列单词,问是否能够连起来. 题目分析:以单词的首尾字母为点,单词为边建立有向图,便是判断图中是否存 ...

  2. Uva 10129 - Play on Words 单词接龙 欧拉道路应用

    跟Uva 10054很像,不过这题的单词是不能反向的,所以是有向图,判断欧拉道路. 关于欧拉道路(from Titanium大神): 判断有向图是否有欧拉路 1.判断有向图的基图(即有向图转化为无向图 ...

  3. UVa 10129 Play On Words【欧拉道路 并查集 】

    题意:给出n个单词,问这n个单词能否首尾接龙,即能否构成欧拉道路 按照紫书上的思路:用并查集来做,取每一个单词的第一个字母,和最后一个字母进行并查集的操作 但这道题目是欧拉道路(下面摘自http:// ...

  4. Nyoj42 一笔画问题 (欧拉道路)

    http://acm.nyist.net/JudgeOnline/problem.php?pid=42题目链接 #include <cstdio> #include <cstring ...

  5. 6-14 Inspector s Dilemma uva12118(欧拉道路)

    题意:给出一个国家城市个数n   所需走过道路个数e   每条道路长t   该国家任意两个城市之间都存在唯一道路长t     要求 :找一条最短的路遍历所有所需走过的路 一开始以为是图的匹配  但是好 ...

  6. POJ 2513 Colored Sticks(欧拉道路+字典树+并查集)

    http://poj.org/problem?id=2513 题意: 给定一些木棒,木棒两端都涂上颜色,求是否能将木棒首尾相接,连成一条直线,要求不同木棒相接的一边必须是相同颜色的. 思路: 题目很明 ...

  7. UVA 10129 Play on Words(欧拉道路)

    题意:给你n个字符串,问你是否可以出现一条链,保证链中每个字符串的第一个元素与上一个字符串的最后一个元素相同,注意可能重复出现同一个字符串 题解:以每一个字符串第一个元素指向最后一个元素形成一个有向图 ...

  8. 【UVa】12118 Inspector's Dilemma(欧拉道路)

    题目 题目     分析 很巧秒的一道题目,对着绿书瞎yy一会. 联一下必须要走的几条边,然后会形成几个联通分量,统计里面度数为奇数的点,最后再减去2再除以2.这样不断相加的和加上e再乘以t就是答案, ...

  9. UVA 10441 - Catenyms(欧拉道路)

    UVA 10441 - Catenyms 题目链接 题意:给定一些单词,求拼接起来,字典序最小的,注意这里的字典序为一个个单词比过去,并非一个个字母 思路:欧拉回路.利用并查集判联通,然后欧拉道路判定 ...

随机推荐

  1. 区间DP 等腰三角形

    题目描述:给定一个正N边形,可以通过连线将这个多边形分割成N-2个三角形,问这N-2个三角形中恰有k个等腰三角形的分割方法有多少?这个值可能很大,输出对9397取模的结果.数据范围:n,k <= ...

  2. mongoDB的复制集5----复制集安全(认证,用户,权限)

    一.什么是认证  如何开启认证    1).auth=true(在配置文件里增加)    2).keyFile(建议添加到配置文件里) #如果设置了auth=true,但第一次没有创建用户就启动实例怎 ...

  3. FINDSTR 命令使用详解

    Findstr 使用正则表达式搜索文件中的文本模式. 语法 findstr [/b] [/e] [/l] [/r] [/s] [/i] [/x] [/v] [/n] [/m] [/o] [/p] [/ ...

  4. MyEclipse10.0的破解过程详细图解

    1 首先下载破解软件包:http://pan.baidu.com/s/1pLB6xEb 并解压: 2 按照百度经验操作即可http://jingyan.baidu.com/article/cbf0e5 ...

  5. HDU 5879 Cure (数论)

    题意:给定n,求前 n 项 1/(k*k) 的和. 析:由于这个极限是 PI * PI / 6,所以我们可以找到分界点,然后计算就好. 代码如下: #pragma comment(linker, &q ...

  6. Vue-i18n实现语言切换

    方法1 Vue — i18n 国际化 全局配置 安 装 1.直接引入js文件 <script src="https://unpkg.com/vue/dist/vue.js"& ...

  7. Gradle系列之一 Groovy语法精讲

    Gradle技术之一 Groovy语法精讲 gradle脚本是基于groovy语言开发的,想要学好gradle必须先要对groovy有一个基本的认识 1. Groovy特点 groovy是一种DSL语 ...

  8. poj 3207 Ikki's Story IV - Panda's Trick【2-SAT+tarjan】

    注意到相交的点对一定要一里一外,这样就变成了2-SAT模型 然后我建边的时候石乐志,实际上不需要考虑这个点对的边是正着连还是反着连,因为不管怎么连,能相交的总会相交,所以直接判相交即可 然后tarja ...

  9. Spring Security 使用Ajax登陆无法跳转页面解决方法

    使用Security的朋友都知道,使用Security后,不再需要我们自己过多的(还需要写少量代码)写登陆的逻辑,只需要自己在html的登陆表单上面定义好输入框name为:username和passw ...

  10. Vue生命周期简介和钩子函数

    钩子就好像是把人的出生到死亡分成一个个阶段,你肯定是在出生阶段起名字,而不会在成年或者死亡的阶段去起名字.或者说你想在出生阶段去约炮,也是不行的.组件也是一样,每个阶段它的内部构造是不一样的.所以一般 ...