【紫书】Play on Words UVA - 10129 欧拉回路

题意:给你1e5个字符串,若前一个的末尾字母等于当前的首字母,则可以连在一起(成语接龙一个意思)判断是否可以将他们连在一起

题解:将首位看作点,单词看作边。变成欧拉回路问题。

    判断出入度是否相等,再用并查集判一下连通性

    (dfs/bfs也行:随便取一个点,搜索一遍。如果每个点都被标记,则是连通的。)

ac代码:

#define _CRT_SECURE_NO_WARNINGS
#include "stdio.h"
#include<stdio.h>
#include<algorithm>
#include<string>
#include<vector>
#include<list>
#include<set>
#include<iostream>
#include<string.h>
#include<queue>
#include<string>
#include<sstream>
#include<stack>
#include<cmath>
using namespace std;
const int maxn = 1e5 + 5;
char s[maxn][1005];
char ss[1005];
int in[256], out[256];
//两个点入度不等于出度且一个差+1,一个-1
int f[256];
int find(int x) {
    return f[x] == x ? x : f[x] = find(f[x]);
}
void un(int x, int y) {
    int u = find(x);
    int v = find(y);
    f[u] = v;
}
bool same(int x,int y) {
    return find(x) == find(y);
}
int main() {
    int t; cin >> t; while (t--) {
        int n; cin >> n;
        for (int i = 0; i < 256; i++) f[i] = i;
        memset(in, 0, sizeof(in)); memset(out, 0, sizeof(out));
        for (int i = 0; i < n; i++) {
            scanf("%s", ss);
            int len = strlen(ss);
            out[ss[0]]++; in[ss[len - 1]]++;
            un(ss[0], ss[len - 1]);

        }
        int ok = 1; //连通
        int yes = 0;//出入度
        vector<int> temp;
        for (int i = ‘a‘; i <= ‘z‘; i++)if(out[i]+in[i]) {//出现过的字母
            if (!same(‘a‘, i)) { ok = 0; break; }
            if (out[i] != in[i]) {
                temp.push_back(i);
            }
        }

        if (temp.size() == 2) {
            int f = temp.front(), b = temp.back();
            if ((abs(in[f] - out[f]) == 1) && (abs(in[b] - out[b]) == 1) && (in[f] + in[b] == out[f] + out[b]))yes=1;
        }
        else if (temp.size() == 0)yes = 1;

        if (ok&&yes)cout << "Ordering is possible.\n";
        else cout << "The door cannot be opened.\n";

    }

    system("pause");
    return 0;
}

原文地址:https://www.cnblogs.com/SuuT/p/8822871.html

时间: 2024-10-04 08:42:13

【紫书】Play on Words UVA - 10129 欧拉回路的相关文章

紫书例题6-3 (UVa 442)

题目地址:https://vjudge.net/problem/UVA-442 题目大意:汗颜,其实我是直接看紫书的中文题意的,大意就是计算两个矩阵乘法次数,设计线性代数知识,可自己百度矩阵乘法. 思路:栈+模拟,左括号开始入栈,右括号开始计算栈顶两个矩阵的乘法次数然后再将新矩阵的n,m入栈即可. AC代码: #include <iostream> #include <string> #include <stack> #include <cstring> u

紫书例题6-4 (UVa 11988)

题目链接:https://vjudge.net/problem/UVA-11988 题目大意:输入一串字符,并按照要求输出,遇到'['字符就将光标移动到开头,遇到']'字符就将光标移动到末尾. 思路: 题目不难懂,很明显的一个模拟就行,重点是如何取存储,这里选择使用链表,链表的具体定义可以去百度看一下,这里不做过多解释,可以简单理解为一个未知长度的数组,它可以借助指针在任意位置插入(删除). 这题具体的操作可以用草稿纸模拟一遍,我是看一位博主的blog明白的,原blog地址:https://bl

【紫书】Oil Deposits UVA - 572 dfs求联通块

题意:给你一个地图,求联通块的数量. 题解: for(所有还未标记的'@'点) 边dfs边在vis数组标记id,直到不能继续dfs. 输出id及可: ac代码: #define _CRT_SECURE_NO_WARNINGS #include "stdio.h" #include<stdio.h> #include<algorithm> #include<string> #include<vector> #include<list&

Play on Words,UVA 10129——求欧拉回路/欧拉通路

Play on words 紫书上的一道题: 输入n(n<=100000)个单词,是否可以把所有这些单词排成一个序列,使得每个单词的第一个字母和上一个单词的最后一个字母相同.每个单词最多包含1000个小写字母.输入中可以有重复单词. 分析: 把字母看成结点,单词看成有向边,则问题有解,当且仅当图中存在欧拉通路/欧拉回路 欧拉通路/回路:通过图中所有边一次且仅一次行遍所有顶点的通路/回路 有向图存在欧拉回路的判断条件:当且仅当有向图是强连通的,且没有奇度顶点,所有顶点的入度等于出度 有向图存在欧拉

第10章例题(紫书)

21/21 题目都很基础,有很多题书上讲得比较详细,然后隔得时间有点久,所以具体什么trick都忘了,思路也懒得去回忆,所以将就着放上来了.... 例题10–1 Uva 11582 题意:输入a, b, n让你计算F[a^b]%n;其中这个F[i]是斐波那契数: 题解: 这题是快速幂+找循环节,用什么方法找循环节呢?因为第一个数是0和1,然后当再出现0和1的时候就是出现循环节的时候,然后假如找到了循环节T,然后就有F[n] = F[n % T],预处理找循环节,O(一百万左右),快速幂logn

第11章例题(紫书)

10/15 这几天先专心刷一下图论的基础题目,也蛮多的,慢慢来... 例题11-1 uva 12219 题意:给你一个表达式,然后又一些子树在之前重复出现过,先要你给这些子树出现的顺序编个号1...N,然后如果重复出现就用编号替代,输出替代之后的表达式. 题解:这是一个表达式树的问题,显示建树,如果让我来写的话,很麻烦,搞不好复杂度是O(n^2),因为字符串是一直扫描下去的,所以就利用一个指针作为全局,然后一直扫下去,就忽略一个左括号,建左树,然后忽略逗号,建右树,忽略右括号,然后一直扫下去,就

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

题意:给出n个单词,问这n个单词能否首尾接龙,即能否构成欧拉道路 按照紫书上的思路:用并查集来做,取每一个单词的第一个字母,和最后一个字母进行并查集的操作 但这道题目是欧拉道路(下面摘自http://blog.csdn.net/hcbbt/article/details/9316301) 关于欧拉道路(from Titanium大神): 判断有向图是否有欧拉路 1.判断有向图的基图(即有向图转化为无向图)连通性,用简单的DFS即可.如果图都不连通,一定不存在欧拉路 2.在条件1的基础上   对于

UAa 1339,紫书P73,词频

题目链接:https://uva.onlinejudge.org/external/13/1339.pdf 紫书P73 解题报告: #include <stdio.h> #include <string.h> #include <stdlib.h> #include <algorithm> using namespace std; int main() { //freopen("input.txt","r",stdin

Play on Words UVA - 10129 欧拉路径

关于欧拉回路和欧拉路径 定义:欧拉回路:每条边恰好只走一次,并能回到出发点的路径欧拉路径:经过每一条边一次,但是不要求回到起始点 ①首先看欧拉回路存在性的判定: 一.无向图每个顶点的度数都是偶数,则存在欧拉回路. 二.有向图(所有边都是单向的)每个节顶点的入度都等于出度,则存在欧拉回路. ②.欧拉路径存在性的判定 一.无向图一个无向图存在欧拉路径,当且仅当   该图所有顶点的度数为偶数   或者  除了两个度数为奇数外其余的全是偶数. 二.有向图一个有向图存在欧拉路径,当且仅当  该图所有顶点的