NYOJ 单词拼接

单词拼接

时间限制: 3000 ms  |  内存限制: 65535 KB
难度: 5
描述

给你一些单词,请你判断能否把它们首尾串起来串成一串。

前一个单词的结尾应该与下一个单词的道字母相同。

aloha

dog

arachnid

gopher

tiger

rat

 

可以拼接成:aloha.arachnid.dog.gopher.rat.tiger

输入
第一行是一个整数N(0<N<20),表示测试数据的组数
每组测试数据的第一行是一个整数M,表示该组测试数据中有M(2<M<1000)个互不相同的单词,随后的M行,每行是一个长度不超过30的单词,单词全部由小写字母组成。
输出
如果存在拼接方案,请输出所有拼接方案中字典序最小的方案。(两个单词之间输出一个英文句号".")
如果不存在拼接方案,则输出
***
样例输入
2
6
aloha
arachnid
dog
gopher
rat
tiger
3
oak
maple
elm
样例输出
aloha.arachnid.dog.gopher.rat.tiger

***

1:判断有向图欧拉图

2 :对单词排序,深搜求最小字典顺,深搜时标记每个边是否使用过

 
#include<bits/stdc++.h>
using namespace std;
#pragma comment(linker,"/STACK:1024000000,1024000000")
const int maxn = 1005*2;
struct node
{
    int to ;
    int order;
};
vector<node>g[maxn];
stack<string>s;
int f[maxn];
int vis[maxn];
int in[maxn];
int out[maxn];
int flag[maxn];
int n;
int  bian[maxn];
string a[maxn];
int res =1;
void init()
{
    for(int i= 0 ; i<=26; i++)
    {
        f[i]  = i;
    }
    for(int i = 0 ; i<n; i++)
        g[i].clear();
    memset(vis,0,sizeof(vis));
    memset(in,0,sizeof(in));
    memset(out,0,sizeof(out));
    memset(flag,0,sizeof(flag));
    memset(bian,0,sizeof(bian));

}
bool  dfs( int u,int cot,int id)
{
    if(cot==n+1)
    {
        return true;

    }
    for(int i = 0 ; i<g[u].size(); i++)
    {
        if(!bian[g[u][i].order])
        {
            bian[g[u][i].order] = 1;
            if(dfs(g[u][i].to,cot+1,g[u][i].order))
            {
                string temp = a[g[u][i].order];
                // cout<<temp <<endl;
                s.push(temp);
                bian[g[u][i].order] = 0;
                //bian[id] = 0 ;
                return true;
            }
            bian[g[u][i].order] = 0;
        }
    }
    return false;
}
int find(int x)
{
    return x==f[x]?x:f[x] = find(f[x]);

}
void add(int x,int y)
{
    int bx = find(x);
    int by = find(y);
    if(bx!=by)
    {
        f[bx] = by;
    }
}
bool judge()
{
    int cot = 0;
    for(int i = 0 ; i<26; i++)
    {
        if(f[i]==i&&flag[i])
            cot++;
    }
    if(cot>1)
        return false;
    return true;
}
int main()
{

    int T;
    scanf("%d",&T);
    while(T--)
    {
        while(!s.empty())
            s.pop();
        res = 0;
        scanf("%d",&n);
        init();
        for(int i = 1 ; i<=n; i++)
        {
            //scanf("%s",a[i]);
            cin>>a[i];
        }

        sort(a+1,a+n+1);
        for(int i = 1 ; i<=n; i++)
        {
            int x = a[i][0]-'a';
            int y = a[i][a[i].size()-1]-'a';
            g[x].push_back((node)
            {
                y,i
            });
            flag[x] = 1;
            flag[y] = 1;
            add(x ,y);
            in[y]++;
            out[x]++;
        }
        int cot = 0;
        int xx,yy;
        for(int i = 0 ; i<26; i++)
        {
            if(in[i]!=out[i])
            {
                cot++;
                if(cot==1)
                {
                    xx = i;
                }
                else if(cot==2)
                {
                    yy = i;
                }
            }
        }

        if(judge())
        {
            if(cot==0||(cot==2&&((in[xx]-out[xx]==1&&out[yy]-in[yy]==1)||(  in[yy]-out[yy]==1&&out[xx]-in[xx]==1)   )))
            {
                // printf("cot = %d\n",cot);
                if(cot==0)
                {
                    if(dfs(a[1][0]-'a',1,1))
                    {
                        while(!s.empty())
                        {
                            cout<<s.top();
                            s.pop();
                            if(!s.empty())
                                cout<<".";
                        }
                    }
                }
                else
                {
                    int jj = 0 ;
                    for(int i = 0 ; i<26&&!jj; i++)
                    {
             
                        if(out[i]-in[i]==1)
                        {
                           
                            if(dfs(i,1,1))
                            {
                       
                                while(!s.empty())
                                {
                                      cout<<s.top();
                                    s.pop();
                                    if(!s.empty())
                                        cout<<".";
                                }
                            }
                        }
                    }
                }
            }
            else
            {
                printf("***");
            }
        }
        else
        {
            printf("***");
        }
        printf("\n");
    }
    return 0 ;
}        


  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
孪生素数是指两个素数之间的差值为2的素数对。通过筛选法可以找出给定素数范围内的所有孪生素数的组数。 在引用的代码中,使用了递归筛选法来解决孪生素数问题。该程序首先使用循环将素数的倍数标记为非素数,然后再遍历素数数组,找出相邻素数之间差值为2的素数对,并统计总数。 具体实现过程如下: 1. 定义一个数组a[N,用来标记数字是否为素数,其中N为素数范围的上限。 2. 初始化数组a,将0和1标记为非素数。 3. 输入要查询的孪生素数的个数n。 4. 循环n次,每次读入一个要查询的素数范围num。 5. 使用两层循环,外层循环从2遍历到num/2,内层循环从i的平方开始,将素数的倍数标记为非素数。 6. 再次循环遍历素数数组,找出相邻素数之间差值为2的素数对,并统计总数。 7. 输出总数。 至此,我们可以使用这个筛选法的程序来解决孪生素数问题。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [python用递归筛选法求N以内的孪生质数(孪生素数)](https://blog.csdn.net/weixin_39734646/article/details/110990629)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] - *2* *3* [NYOJ-26 孪生素数问题](https://blog.csdn.net/memoryofyck/article/details/52059096)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值