前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >10106. 「一本通 3.7 例 2」单词游戏

10106. 「一本通 3.7 例 2」单词游戏

作者头像
yzxoi
发布2022-09-19 12:05:40
3300
发布2022-09-19 12:05:40
举报
文章被收录于专栏:OI

10106. 「一本通 3.7 例 2」单词游戏

题意

N 个盘子,每个盘子上写着一个仅由小写字母组成的英文单词。你需要给这些盘子安排一个合适的顺序,使得相邻两个盘子中,前一个盘子上单词的末字母等于后一个盘子上单词的首字母。请你编写一个程序,判断是否能达到这一要求。如果能,请给出一个合适的顺序。多组数据。第一行给出数据组数 T,每组数据第一行给出盘子数量 N,接下去 N 行给出小写字母字符串,一种字符串可能出现多次。若存在一组合法解输出Ordering is possible.,否则输出 The door cannot be opened.。

思路

很明显就是寻找有没有欧拉图。 先来看一看欧拉图的定义:

有向图G为半欧拉图,当且仅当G为连通图,且存在顶点u的入度比出度大1,v的入度比出度小1,其它所有顶点的入度等于出度。 存在欧拉路径而不存在欧拉回路。

再来看一看有向图的欧拉图的性质: 1.图G是连通的,不能有孤立的点存在。 2.存在两个顶点,其入度不等于出度,其中一点出度比入度大1,为路径起点,另一点入度比出度大1,为路径的终点。 好了,有了这些信息,那么这道题就很好打了。 可以采用并查集来帮助check是否为半欧拉图。

代码语言:javascript
复制
#include<algorithm>
#include<bitset>
#include<complex>
#include<deque>
#include<exception>
#include<fstream>
#include<functional>
#include<iomanip>
#include<ios>
#include<iosfwd>
#include<iostream>
#include<istream>
#include<iterator>
#include<limits>
#include<list>
#include<locale>
#include<map>
#include<memory>
#include<new>
#include<numeric>
#include<ostream>
#include<queue>
#include<set>
#include<sstream>
#include<stack>
#include<stdexcept>
#include<streambuf>
#include<string>
#include<typeinfo>
#include<utility>
#include<valarray>
#include<vector>
#include<cstring>
using namespace std;
#define MAXE 400010
inline int read(){
    char ch=getchar();int res=0,f=1;
    while(ch<'0'ch>'9'){if(ch=='-') f=-1;ch=getchar();}
    while(ch>='0'&&ch<='9') res=res*10+ch-'0',ch=getchar();
    return res*f;
}
int type,n,m;
int in[MAXE],out[MAXE],fa[MAXE];
bool vis[MAXE];
int T;
char as[1010];
int getfa(int x){
    return x==fa[x]?x:fa[x]=getfa(fa[x]);
}
int main(){
    T=read();
    while(T--){
        int Fans=0;
        n=read();m=n;
        memset(in,0,sizeof(in));
        memset(out,0,sizeof(out));
        for(int i=1;i<=26;i++) fa[i]=i;
        for(int i=1;i<=n;i++){
            int x,y;
            cin>>as;
            x=as[0]-'a'+1,y=as[strlen(as)-1]-'a'+1;
            fa[getfa(y)]=getfa(x);
            in[y]++;out[x]++;
        }
        int ans=0;
        for(int i=1;i<=26;i++){
            if(((in[i]!=0out[i]!=0)&&(getfa(i)==i))abs(in[i]-out[i])>1) ans++;//root and son >=2 
        }
        if(ans>=2){puts("The door cannot be opened.");continue;}
        int ans1=0,ans2=0;
        for(int i=1;i<=26;i++) (in[i]>out[i]?ans1++:(in[i]<out[i]?ans2++:ans));
        if(ans1!=ans2ans1>1) puts("The door cannot be opened.");
        else puts("Ordering is possible.");
    }
    return 0;
}
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2019-02-15 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 10106. 「一本通 3.7 例 2」单词游戏
    • 题意
      • 思路
      领券
      问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档