题意:给你n(n<=10000)个字符串,每个字符串的长度不超过30,可以选择两个非空前缀把它们拼起来得到一个字符串(这两个前缀可以来自同一个字符串,也可以是同一个字符串的同一个非空前缀),问得到的所有字符串中有多少个本质不同的字符串.

首先看到一堆字符串的前缀我们就可以想AC自动机,这个题意看上去只要在AC自动机上DP一下就好了,然后我看了眼题解发现确实是AC自动机上DP,然后就开始想怎么DP,想了2h才搞出来...交一发T了,减少无用状态的转移A了,15s,榜上倒数第二...(后来把指针+动态内存改为指针+静态内存,18s了...目测是大数组申请和构造函数的锅?)

其实我虽然搞出来了但是并不能讲明白

那么怎么DP呢?首先我们需要把一个可行的字符串对应为AC自动机上从根节点出发的一条路径,然后有两种情况,一种是这条路径没有沿着fail指针往回走过,那么这条路径本身也对应着某个串的一个前缀,我们在AC自动机上遍历一遍(实现时可以在求fail指针时顺便做)就可以统计所有的这种路径,判断这条路径能否拆分成两个前缀就只需看路径的终点的fail指针是否指向根节点即可(如果不指向根节点,那么fail指针指向的那个节点对应了拆分方案中后面的那个前缀,而去掉后面那个前缀之后,这条路径前面的部分必然还是一个前缀).

第二种情况就是这条路径沿着fail指针往回走过. 这种情况比上一种情况复杂.如何判断一条路径是否合法?我们可以在这条路径的开头找出一个尽量长的前缀,然后在这条路径的结尾找出一个尽量长的前缀,判断这样的两个前缀能否组成整条路径.那么开头位置的尽量长的前缀对应着从AC自动机的根节点走到这条路径第一次跳fail指针的位置.假如在第一次跳fail指针后走了x步到达了终点,那么终点的深度对应着这条路径结尾位置的最长前缀.只要终点的深度大于等于x,我们就能找到合法的拆分方案把这条路径拆成两个前缀.

那么定义状态时首先可以想到f[i][j][k]表示从第i个节点第一次跳fail指针,走了j步到达一个深度为k的节点的方案有多少,这样好像会MLE+TLE.注意到从哪个位置开始第一次跳fail指针并没有什么用,我们关注的是终点的深度,那么定义f[i][j]表示第一次跳fail指针之后走了j步到达节点i的方案有多少,我写的复杂度是O(节点总数*最大深度*字符集大小),也就是300000×30×26....减少一些无用状态的转移之后能15s跑过去也是感人....榜上大神们都跑得好快呀不知道是有复杂度更好的方法还是我的常数太丑了?不过加了滚动数组之后10s了233333

#include<cstdio>
#include<cstring>
#include<queue>
#include<algorithm>
using namespace std;
const int maxn=;
struct node{
node* ch[],*fail;
int num,depth;
node(){}
node(int x,int d){depth=d;memset(ch,,sizeof(ch));fail=;num=x;}
}*root;int tot=;
char str[];
node* pos[maxn];
void Add(char *c){
node* p=root;
while(*c){
int t=*c-'a';
if(p->ch[t]==NULL){p->ch[t]=new node(++tot,p->depth+);pos[tot]=p->ch[t];}
p=p->ch[t];++c;
}
}
long long f[][maxn];
long long ans=;
void getfail(){
queue<node*> q;q.push(root);
while(!q.empty()){
node* x=q.front();q.pop();
if(x!=root&&x->fail!=root)ans++;
for(int i=;i<;++i){
if(x->ch[i]){
if(x==root)x->ch[i]->fail=root;
else x->ch[i]->fail=x->fail->ch[i];
q.push(x->ch[i]);
}else{
if(x==root)x->ch[i]=root;
else x->ch[i]=x->fail->ch[i];
f[][x->ch[i]->num]++;
}
}
}
}
int main(){
int n;scanf("%d",&n);int maxlen=;
root=new node(,);pos[]=root;
for(int i=;i<=n;++i){
scanf("%s",str);int len=strlen(str);if(len>maxlen)maxlen=len;
Add(str);
}
getfail();
int flag=;
for(int k=;k<=maxlen;++k){
for(int i=;i<=tot;++i)f[flag^][i]=;
for(int i=;i<=tot;++i){
if(f[flag][i]==||pos[i]->depth<k)continue;
for(int j=;j<;++j){
f[flag^][pos[i]->ch[j]->num]+=f[flag][i];
}
if(k<=pos[i]->depth)ans+=f[flag][i];
//printf("f[%d][%d]==%lld\n",i,k,f[i][k]);
}
flag^=;
}
printf("%lld\n",ans);
return ;
}
#include<cstdio>
#include<cstring>
#include<queue>
#include<algorithm>
using namespace std;
const int maxn=;
struct node{
node* ch[],*fail;
int num,depth;
node(int x,int d){depth=d;memset(ch,,sizeof(ch));fail=;num=x;}
}*root;int tot=;
char str[];
node* pos[maxn];
void Add(char *c){
node* p=root;
while(*c){
int t=*c-'a';
if(p->ch[t]==NULL){p->ch[t]=new node(++tot,p->depth+);pos[tot]=p->ch[t];}
p=p->ch[t];++c;
}
}
long long f[maxn][];
long long ans=;
void getfail(){
queue<node*> q;q.push(root);
while(!q.empty()){
node* x=q.front();q.pop();
if(x!=root&&x->fail!=root)ans++;
for(int i=;i<;++i){
if(x->ch[i]){
if(x==root)x->ch[i]->fail=root;
else x->ch[i]->fail=x->fail->ch[i];
q.push(x->ch[i]);
}else{
if(x==root)x->ch[i]=root;
else x->ch[i]=x->fail->ch[i];
f[x->ch[i]->num][]++;
}
}
}
}
int main(){
int n;scanf("%d",&n);int maxlen=;
root=new node(,);pos[]=root;
for(int i=;i<=n;++i){
scanf("%s",str);int len=strlen(str);if(len>maxlen)maxlen=len;
Add(str);
}
getfail();
// for(int i=1;i<=tot;++i){
// for(int j=0;j<26;++j){
// if(pos[i]->ch[j]->num<=i){
// f[pos[i]->ch[j]->num][1]++;
// }
// }
// }
//printf("ans==%lld\n",ans);
for(int k=;k<=maxlen;++k){
for(int i=;i<=tot;++i){
if(f[i][k]==||pos[i]->depth<k)continue;
for(int j=;j<;++j){
f[pos[i]->ch[j]->num][k+]+=f[i][k];
}
if(k<=pos[i]->depth)ans+=f[i][k];
//printf("f[%d][%d]==%lld\n",i,k,f[i][k]);
}
}
printf("%lld\n",ans);
return ;
}

bzoj4502 串的更多相关文章

  1. BZOJ4502串——AC自动机(fail树)

    题目描述 兔子们在玩字符串的游戏.首先,它们拿出了一个字符串集合S,然后它们定义一个字 符串为“好”的,当且仅当它可以被分成非空的两段,其中每一段都是字符串集合S中某个字符串的前缀. 比如对于字符串集 ...

  2. ASP.NET MVC5+EF6+EasyUI 后台管理系统(62)-EF链接串加密

    系列目录 前言: 这一节提供一个简单的功能,这个功能看似简单,找了一下没找到EF链接数据库串的加密帮助文档,只能自己写了,这样也更加符合自己的加密要求 有时候我们发布程序为了避免程序外的SQL链接串明 ...

  3. JQuery使用deferreds串行多个ajax请求

    使用JQuery对多个ajax请求串行执行. HTML代码: <a href="#">Click me!</a> <div></div&g ...

  4. iOS 字典或者数组和JSON串的转换

    在和服务器交互过程中,会iOS 字典或者数组和JSON串的转换,具体互换如下: // 将字典或者数组转化为JSON串 + (NSData *)toJSONData:(id)theData { NSEr ...

  5. iOS:GCD理解1(同步-异步、串行-并行)

    1.并行-异步(ST1与ST2抢占资源) 1-1).获取 并行(全局)队列 ,DISPATCH_QUEUE_PRIORITY_DEFAULT 为默认优先级. dispatch_queue_t queu ...

  6. 关于用sql语句实现一串数字位数不足在左侧补0的技巧

    在日常使用sql做查询插入操作时,我们通常会用到用sql查询一串编号,这串编号由数字组成.为了统一美观,我们记录编号时,统一指定位数,不足的位数我们在其左侧补0.如编号66,我们指定位数为5,则保存数 ...

  7. [LeetCode] Longest Palindrome 最长回文串

    Given a string which consists of lowercase or uppercase letters, find the length of the longest pali ...

  8. [LeetCode] Shortest Palindrome 最短回文串

    Given a string S, you are allowed to convert it to a palindrome by adding characters in front of it. ...

  9. [LeetCode] Palindrome Partitioning II 拆分回文串之二

    Given a string s, partition s such that every substring of the partition is a palindrome. Return the ...

随机推荐

  1. 接口测试工具postman(五)批量执行测试用例

    1.准备好测试用例及相关数据 2.点击Run按钮 3.选择运行collection或者folder 4.运行完成

  2. MVC数据的注册及验证简单总结

    一.注解 注解是一种通用机制,可以用来向框架注入元数据,同时,框架不只驱动元数据的验证,还可以在生成显示和编辑模型的HTML标记时使用元数据. 二.验证注册的使用 1.Require:属性为Null或 ...

  3. java 实现redis缓存

    由于项目加载时请求数据量过大,造成页面加载很慢.采用redis作缓存,使二次访问时页面,直接取redis缓存. 1.redis连接参数 2.连接redis,设置库 3.配置文件开启缓存 4.mappe ...

  4. 小程序开发中,纯css实现内容收起折叠功能

    不多说,直接上代码: wxml页面: <!--收起折叠 begin--> <view style='width:100%;background:#fff;border-top:1px ...

  5. Windows10安装GPU版本的Tensorflow

    本人电脑配置(公司的)gtx1080ti,下载的的cuda8.0,cudnn6.0,python3.5.3安装完成后,安装tensorflow 1.pip install tensorflow-gpu ...

  6. python3爬虫-快速入门-爬取图片和标题

    直接上代码,先来个爬取豆瓣图片的,大致思路就是发送请求-得到响应数据-储存数据,原理的话可以先看看这个 https://www.cnblogs.com/sss4/p/7809821.html impo ...

  7. [译] JavaScript核心指南(JavaScript Core) 【转】

    本文转自:http://remember2015.info/blog/?p=141#scope-chain 零.索引 对象(An Object) 原型链(A Prototype Chain) 构造函数 ...

  8. Elasticsearch 评分score计算中的Boost 和 queryNorm

    本来没有这篇文章,在公司分享ES的时候遇到一个问题,使用boost的时候,怎么从评分score中知道boost的影响. 虽然我们从查询结果可以直观看到,boost起了应有的作用,但是在explain的 ...

  9. Linux内核设计笔记10——内核同步

    Linux内核同步笔记 几个基本概念 - 临界区(critical region):访问和操作共享数据的代码段: - 原子操作:操作在执行中不被打断,要么不执行,要么执行完: - 竞争条件: 两个线程 ...

  10. ThinkPHP - 2 - SAE(新浪云)部署

    ThinkPHP3.2核心内置了对SAE平台的支持(采用了应用模式的方式),具有自己的独创特性,能够最大程度的使用ThinkPHP的标准特性,让开发人员感受不到SAE和普通环境的差别.甚至可以不学习任 ...