题目链接


Solution

感觉比较巧的题啊...

考虑几点:

  • 可以交换无数次字母表,即字母表可以为任意形态.

  • 对于以其他字符串为前缀的字符串,我们可以直接舍去.

    因为此时它所包含的前缀的字典序绝对比它本身小.

  • 需要使得某个字符串 \(S\) 字典序最小,需要讨论两种情况:

    \(1.\) 与它没有公共前缀的字符串

    此时我们即使得 \(S_{1}\) 大于其第一个即可.

    \(2.\) 与它有公共前缀的字符串

    我们令其最长公共前缀的位置为 \(k\) .

    那么此时我们即要求,对于任意字符串 \(T\), 在字母表中 \(S_{k+1}<T_{k+1}\) .

    我们可以将这种关系在 \(26\) 个字母中形成一张关系图,判断是否满足条件.

    直接判环即可.

其实以上两种情况可以都理解为第二种情况,只是第一种情况的\(k=0\)罢了.

怎样去快速定位公共前后缀的位置?

我们使用 \(Trie\) 树,直接将所有字符串插入 \(Trie\) 树中.

每一次处理都对当前节点所有的非本字符串节点连边,然后拓扑排序判环即可.

总时间复杂度 \(O(sum_{len}*26)\)


Code

#include<bits/stdc++.h>
using namespace std;
const int maxn=300008;
int ch[maxn][26];
int num[maxn],n,cnt;
struct sj{
int to;
int next;
}a[1008];
int du[30],tot,fuck;
int head[30],size;
string ans[maxn]; void add(int x,int y)
{
a[++size].to=y;
a[size].next=head[x];
head[x]=size;
} void dfs()
{
queue<int>q;
for(int i=0;i<26;i++)
if(!du[i])
q.push(i);
while(!q.empty())
{
int now=q.front();q.pop();
for(int i=head[now];i;i=a[i].next)
{
int tt=a[i].to;
du[tt]--;
if(!du[tt])
q.push(tt);
}
}
} int insert(char *s)
{
int len=strlen(s),u=0;
for(int i=0;i<len;i++)
{
if(!ch[u][s[i]-'a'])
ch[u][s[i]-'a']=++tot;
if(num[ch[u][s[i]-'a']])return 1;
u=ch[u][s[i]-'a'];
}
num[u]++;
return 0;
} int pre(string s)
{
int len=s.length(),u=0;
for(int i=0;i<len;i++)
{
int p=s[i]-'a'; for(int j=0;j<26;j++)
{
if(ch[u][j]!=0&&j!=p)
add(p,j),
du[j]++;
}
u=ch[u][p];
if(i!=len-1&&num[u]!=0)return 0;
}
return 1;
}
string s[maxn];
char cha[maxn];
int main()
{
scanf("%d",&n);
for(int i=1;i<=n;i++)
{
scanf("%s",cha);
if(!insert(cha))
{
cnt++;
int len=strlen(cha);
for(int j=0;j<len;j++)
s[cnt]+=cha[j];
}
}
for(int i=1;i<=cnt;i++)
{
memset(a,0,sizeof(a)),size=0;
memset(head,0,sizeof(head));
memset(du,0,sizeof(du));
if(!pre(s[i])){continue;}
dfs();
int flag=1;
for(int j=0;j<26;j++)
if(du[j])flag=0; if(flag)
ans[++fuck]=s[i];
}
cout<<fuck<<endl;
for(int i=1;i<=fuck;i++)
cout<<ans[i]<<endl;
}

P3065 [USACO12DEC]第一!First!

[USACO12DEC]第一!First! (Trie树,拓扑排序)的更多相关文章

  1. 洛谷P3065 [USACO12DEC]第一!First!(Trie树+拓扑排序)

    P3065 [USACO12DEC]第一!First! 题目链接:https://www.luogu.org/problemnew/show/P3065 题目描述 Bessie一直在研究字符串.她发现 ...

  2. 「Usaco2012 Dec」第一(字典树+拓扑排序)

    (我恨字符串) 惯例化简题目:给定n个字符串,可以改变字符的相对大小(在字典序中的大小),问:字符串i是否能成为最小的字符串(字典序) 解题过程: 首先你可以预处理出来26的全排列然后暴力然后你只要用 ...

  3. Luogu P3065 [USACO12DEC]第一!First!【字典树/拓扑排序】By cellur925

    题意:给你许多字符串,你可以改变字母序大小,问有哪些字符串可能成为字典序最小的字符串. 我们考虑把这些字符串都塞到\(trie\)树上.之后检索每一个字符串的时候,我们看和他同一层的地方是否有字符,如 ...

  4. BZOJ_3012_[Usaco2012 Dec]First!_trie树+拓扑排序

    BZOJ_3012_[Usaco2012 Dec]First!_trie树+拓扑排序 题意: 给定n个总长不超过m的互不相同的字符串,现在你可以任意指定字符之间的大小关系.问有多少个串可能成为字典序最 ...

  5. 牛客练习赛11 假的字符串 (Trie树+拓扑找环)

    牛客练习赛11 假的字符串 (Trie树+拓扑找环) 链接:https://ac.nowcoder.com/acm/problem/15049 来源:牛客网 给定n个字符串,互不相等,你可以任意指定字 ...

  6. [bzoj3012][luogu3065][USACO12DEC][第一!First!] (trie+拓扑排序判环)

    题目描述 Bessie has been playing with strings again. She found that by changing the order of the alphabe ...

  7. 【BZOJ-2938】病毒 Trie图 + 拓扑排序

    2938: [Poi2000]病毒 Time Limit: 1 Sec  Memory Limit: 128 MBSubmit: 609  Solved: 318[Submit][Status][Di ...

  8. BZOJ3832[Poi2014]Rally——权值线段树+拓扑排序

    题目描述 An annual bicycle rally will soon begin in Byteburg. The bikers of Byteburg are natural long di ...

  9. BZOJ4383 Pustynia(线段树+拓扑排序)

    线段树优化建图暴力拓扑排序即可.对于已确定的数,拓扑排序时dp,每个节点都尽量取最大值,如果仍与已确定值矛盾则无解.叶子连出的边表示大于号,其余边表示大于等于. #include<iostrea ...

随机推荐

  1. [论文理解] Connectionist Text Proposal Network

    Connectionist Text Proposal Network 简介 CTPN是通过VGG16后在特征图上采用3*3窗口进行滑窗,采用与RPN类似的anchor机制,固定width而只预测an ...

  2. 如何使用Python生成200个优惠券(激活码)

    解析: 常见的优惠券(激活码)是由数字.字母(大小写)组成: string.ascii_letters   26个大小写字母: string.digits 0-9数字: 随机组合 使用random.s ...

  3. CMDB数据库设计

    title: CMDB 数据库设计 tags: Django --- CMDB数据库设计 具体的资产 服务器表和网卡.内存.硬盘是一对多的关系,一个服务器可以有多个网卡.多个内存.多个硬盘 hostn ...

  4. bp神经网络原理

    bp(back propagation)修改每层神经网络向下一层传播的权值,来减少输出层的实际值和理论值的误差 其实就是训练权值嘛 训练方法为梯度下降法 其实就是高等数学中的梯度,将所有的权值看成自变 ...

  5. (72)zabbix监控日志文件 MySQL日志为例

    一般情况下,日志最先反映出应用当前的问题,在海量日志里面找到我们异常记录,然后记录下来,并且根据情况报警,大家可以监控系统日志.nginx.Apache.业务日志. 这边我拿常见的MySQL日志做监控 ...

  6. Python9-进程理论-day35

    #!/usr/bin/env python# -*- coding:utf-8 -*-# Author:Tim'''进程(Process)是计算机中的程序关于某数据集合上的一次运行活动,是系统进行资源 ...

  7. (转)可简化iOS 应用程序开发的6个Xcode小技巧

    Xcode是iPhone和iPad开发者用来编码或者开发iOS app的IDE.Xcode有很多小巧但很有用的功能,很多时候我们可能没有注意到它们,也或者我们没有在合适的水平使用这些功能简化我们的iO ...

  8. AOP面向切面编程笔记

    1.AOP概念:Aspect Oriented Programming 面向切面编程 2.作用:本质上来说是一种简化代码的方式 继承机制 封装方法 动态代理 …… 3.情景举例 ①数学计算器接口[Ma ...

  9. SQL前后端分页

    /class Page<T> package com.neusoft.bean; import java.util.List; public class Page<T> { p ...

  10. 学习pwn的一些指导

    使用ret2libc攻击方法绕过数据执行保护 http://blog.csdn.net/linyt/article/details/43643499 格式化字符串利用小结 http://www.cnb ...