题目链接:HDU-5129

题目大意为给一堆字符串,问由任意两个字符串的前缀子串(注意断句)能组成多少种不同的字符串。

思路是先用总方案数减去重复的方案数。

考虑对于一个字符串S,如图,假设S1,S2,S3,S4,S5,S6均为前缀。

换言之,对于这种字符串,我们计算了三次。

发现,重复的方案数,等于中间如图有颜色的方块的数量。所以我们要做的也就是计数像图中有颜色的小方块的数量。

我们可以通过遍历像S6一样的字符串的数量,来计算重复的方案数。S6满足以下条件:

  1. 存在一个前缀S4为S6的后缀,且S4为S6的最长的“是前缀的”后缀(保证是小方块间没有隔板)。
  2. S6-S4为某个前缀的后缀(在图中为S3的后缀)。

我们发现,对于某一个S6,其对应的S4时一定的。

我们用sum[S]表示以S为后缀的前缀字符串的数量。

这样,当我们遍历S6,对于每一个S6,我们找到其对应的S4后,只要减去( sum[S6-S4] - 1 )即可。

所以我们要处理的,就是以下两个问题:

  1. 如何找到S6对应的S4
  2. 如何求sum[S]

这时候,我们发现S6与S4的关系,和AC自动机的性质很像。S6在失配时跳到的位置就是S4。于是第一个问题解决了。

对于第二个问题,我们同样可以通过AC自动机找到。具体的参见代码。

 #include<cstring>
#include<cstdio>
#include<queue>
using namespace std; typedef long long LL;
const LL MAXN=;
const LL SIGMA_SIZE=;
struct Trie
{
LL ch[MAXN][SIGMA_SIZE];
LL fa[MAXN];
LL sz; //节点总数
Trie() { sz=; fa[]=-; memset(ch[],,sizeof(ch[])); }
LL idx(char c) { return c-'a'; } //节点c的编号
void clear() { sz=; fa[]=-; memset(ch[],,sizeof(ch[])); } //在Trie中插入字符串s
void insert(char *s)
{
LL u=,n=strlen(s);
for(LL i=;i<n;i++)
{
LL c=idx(s[i]);
if(!ch[u][c]) //节点不存在
{
memset(ch[sz],,sizeof(ch[sz]));
fa[sz]=u;
ch[u][c]=sz++; //新建节点
}
u=ch[u][c]; //往下走
}
} //AC自动机部分
LL f[MAXN];
LL deg[MAXN];
LL sum[MAXN]; //sum[i]表示以Si为后缀的前缀的数量
void getFail()
{
queue<LL> q;
f[]=;
//初始化队列
for(LL c=;c<SIGMA_SIZE;c++)
{
LL u=ch[][c];
if(u) { f[u]=; q.push(u); }
}
//按BFS顺序计算失配函数
while(!q.empty())
{
LL r=q.front(); q.pop();
for(LL c=;c<SIGMA_SIZE;c++)
{
LL u=ch[r][c];
if(!u) { ch[r][c]=ch[f[r]][c]; continue; }
q.push(u);
LL v=f[r];
f[u]=ch[v][c];
}
}
for(LL i=;i<sz;i++) { deg[i]=; sum[i]=; }
for(LL i=;i<sz;i++) deg[f[i]]++;
queue<LL> Q;
for(LL i=;i<sz;i++) if(!deg[i]) Q.push(i);
while(!Q.empty())
{
LL u=Q.front(); Q.pop();
sum[f[u]]+=sum[u];
deg[f[u]]--;
if(!deg[f[u]]) Q.push(f[u]);
}
} void solve()
{
LL tot=;
for(LL i=;i<sz;i++) if(f[i])
{
LL j=f[i];
LL p=i;
while(j)
{
p=fa[p];
j=fa[j];
}
tot+=sum[p]-;
}
printf("%lld\n",1LL*(sz-)*(sz-)-tot);
}
};
Trie T;
int main()
{
#ifdef LOCAL
freopen("in.txt","r",stdin);
#endif
LL n;
while(scanf("%lld",&n) && n)
{
scanf("%lld",&n);
for(LL i=;i<=n;i++)
{
char s[];
scanf("%s",s);
T.insert(s);
}
T.getFail();
T.solve();
T.clear();
}
return ;
}

HDU 5129 Yong Zheng's Death的更多相关文章

  1. 字符串(AC自动机):HDU 5129 Yong Zheng's Death

    Yong Zheng's Death Time Limit: 20000/10000 MS (Java/Others)    Memory Limit: 512000/512000 K (Java/O ...

  2. hdu 5129 (枚举) The E-pang Palace

    题目;http://acm.hdu.edu.cn/showproblem.php?pid=5128. 给你n个点,问能否组成两个不相交的与坐标轴平行的矩形,能就输出两矩形的面积和,不能就输出一个字符串 ...

  3. 2014ACM/ICPC亚洲区广州站题解

    这一场各种计算几何,统统没有做. HDU 5129 Yong Zheng's Death HDU 5136 Yue Fei's Battle

  4. 推荐系统(1)--splitting approaches for context-aware recommendation

    开篇语: 大一的时候.在实验室老师和师兄的带领下.我開始接触推荐系统.时光匆匆,转眼已是大三,因为大三课甚是少.于是便有了时间将自己所学的东西做下总结. 第一篇博客.献给过去三年里带我飞的老师和师兄们 ...

  5. HDU 5860 Death Sequence(死亡序列)

    p.MsoNormal { margin: 0pt; margin-bottom: .0001pt; text-align: justify; font-family: Calibri; font-s ...

  6. HDU 5860 Death Sequence(递推)

    HDU 5860 Death Sequence(递推) 题目链接http://acm.split.hdu.edu.cn/showproblem.php?pid=5860 Description You ...

  7. HDU 6136 Death Podracing(循环链表)

    [题目链接] http://acm.hdu.edu.cn/showproblem.php?pid=6136 [题目大意] 一堆人在操场上跑步,他们都有一定的速度和初始位置, 当两个人相遇的时候编号较小 ...

  8. 2016 Multi-University Training Contest 10 || hdu 5860 Death Sequence(递推+单线约瑟夫问题)

    题目链接:http://acm.split.hdu.edu.cn/showproblem.php?pid=5860 题目大意:给你n个人排成一列编号,每次杀第一个人第i×k+1个人一直杀到没的杀.然后 ...

  9. HDU 6136 Death Podracing (堆)

    题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=6136 题解 完了,普及题都不会做了... 发现一个重要性质是只有相邻的人才会相撞,于是直接拿堆维护即可 ...

随机推荐

  1. 转:DP和HDP

    Dirichlet Process and Hierarchical Dirichlet Process 原文:http://hi.baidu.com/zentopus/item/46a622f5ef ...

  2. CentOS yum安装软件包

    yum(Yellowdog Update Modifie)命令是在Fedora和RedHat以及SUSE中基于rpm的软件包管理器,它可以使系统管理人员交互和自动化地更细与管理RPM软件包,能够从指定 ...

  3. 一条数据的HBase之旅,简明HBase入门教程-Write全流程

    如果将上篇内容理解为一个冗长的"铺垫",那么,从本文开始,剧情才开始正式展开.本文基于提供的样例数据,介绍了写数据的接口,RowKey定义,数据在客户端的组装,数据路由,打包分发, ...

  4. HTTPS证书生成方法,也适用于APP

    步骤: 一.登录服务器,创建ioscert目录 cd ~/servers/APP mkdir ioscert 二.cd到ioscert目录下,执行以下命令: 1.openssl genrsa -out ...

  5. 函数strcpy的实现

    strcpy函数的百科中给出了各种情况的详细说明,这里,仅给出一些注意事项: 1.strcpy的函数原型是: /* dest(destination)为目标字符串,src(source)为原字符串*/ ...

  6. Codeforces VK Cup Finals #424 Div.1 A. Office Keys(DP)

    显然是不可能交叉取钥匙的,于是把钥匙和人都按坐标排序就可以DP了 钥匙可以不被取,于是f[i][j]表示前i个钥匙被j个人拿的时间 f[i][j]=min(f[i-1][j],max(f[i-1][j ...

  7. bzoj1901: Zju2112 Dynamic Rankings(BIT套主席树)

    带修改的题主席树不记录前缀,只记录单点,用BIT统计前缀.  对于BIT上每一个点建一棵主席树,修改和询问的时候用BIT跑,在主席树上做就行了.  3k4人AC的题#256...应该不算慢 #incl ...

  8. BAT-Python面试题

    Python语言特性 1 Python的函数参数传递 看两个如下例子,分析运行结果: 代码一: a = 1 def fun(a): a = 2 fun(a) print(a) # 1 代码二: a = ...

  9. 如何更有效使用 Rational AppScan 扫描大型网站,第 2 部分: 案例分析

    使用 AppScan 进行扫描 针对大型网站的扫描,我们按照戴明环 PDCA 的方法论来进行规划和讨论,建议 AppScan 使用步骤:计划(Plan).执行(Do).检查(check).分析(Ana ...

  10. overflow:auto产生的滚动条在安卓系统下能平滑滚动,而在ios下滚动不平滑

    由于系统的问题,加上-webkit-overflow-scrolling : touch; 即可解决平滑滚动问题