Portal --> bzoj4212

Description

​  给你\(n\)个字符串,接下来有\(m\)个询问,每个询问由两个给定的字符串\(s_1\)和\(s_2\)组成,对于每个询问输出\(n\)个字符串中有多少个同时满足\(s_1\)是其前缀且\(s_2\)是其后缀,强制在线

​​  数据范围:\(n<=2000,m<=100000\),\(n\)个字符串总长度以及询问字符串总长度均\(<=2000000\)

​  

Solution

​​  (怎么感觉自己好像没有学过trie一样是因为太久没有写所以根本没有往那个方向想吗==)

​  最暴力的做法的话。。想法很简单:先确定哪些字符串满足\(s_1\)是前缀,再看其中满足\(s_2\)是后缀的数量

​  前缀这种东西的话,如果说我们将\(n\)个字符串丢到trie里面,将dfs一遍之后各个字符串结尾对应节点的访问顺序存在一个数组里面,那么对于一个询问\((s_1,s_2)\),满足\(s_1\)是前缀的字符串的结尾对应节点必定在一个子树当中,反应到数组里面的话就是一个区间

​​  那所以接下来我们要做的就是查这个区间内的字符串有多少个满足\(s_2\)是后缀,只要将所有的串反转然后丢到trie里面,就又转成查前缀的问题了,解决方式和上面一样

​​  至于区间这个问题,直接可持久化一下就好了

​  

​  mark:**没事还是不要随便用getline好了==评测的时候超级容易翻车 **

​​  mark:前缀什么的也是应该考虑一下trie啊qwq不要看到字符串什么的就满脑子sa、sam之类的

​  

​  代码大概长这个样子

#include<iostream>
#include<cstdio>
#include<cstring>
#include<vector>
#define mp make_pair
#define Pr pair<int,int>
using namespace std;
const int N=2010,L=2000010,C=26;
string s[N],s1,s2;
char tmps[L];
int st[N],lis[N];
int n,tot,lastans,m;
namespace T1{/*{{{*/
const int N=::L;
int ch[N][C],st[N],ed[N];
vector<int> mark[N];
int tot,rt;
void init(){rt=1;tot=1;}
void insert(string s,int id){
int len=s.length(),now=rt,c;
for (int i=0;i<len;++i){
c=s[i]-'a';
if (!ch[now][c]) ch[now][c]=++tot;
now=ch[now][c];
}
mark[now].push_back(id);
}
void dfs(int x){
bool have=mark[x].size();
if (mark[x].size()){
st[x]=lis[0]+1;
for (int i=0;i<mark[x].size();++i)
lis[++lis[0]]=mark[x][i];
}
bool flag=false;
for (int i=0;i<C;++i){
if (!ch[x][i]) continue;
dfs(ch[x][i]);
if (!flag&&!have)
st[x]=st[ch[x][i]];
flag=true;
}
ed[x]=lis[0];
}
void get_order(){lis[0]=0; dfs(rt);}
Pr query(string s){
int now=rt,len=s.length(),c;
for (int i=0;i<len;++i){
//printf("%c\n",s[i]);
c=(s[i]-'a'+lastans)%C;
now=ch[now][c];
}
if (!now) return mp(-1,-1);
return mp(st[now],ed[now]);
}
}/*}}}*/
namespace T2{/*{{{*/
const int N=::L;
int ch[N][C],rt[::N],cnt[N];
int tot;
void init(){tot=0;rt[0]=0;}
int newnode(int pre){
cnt[++tot]=cnt[pre];
for (int i=0;i<C;++i) ch[tot][i]=ch[pre][i];
return tot;
}
void _insert(int pre,int &x,string &s,int loc){
x=newnode(pre);
++cnt[x];
if (loc<0) return;
int c=(s[loc]-'a'+lastans)%C;
_insert(ch[pre][c],ch[x][c],s,loc-1);
}
void insert(int pre,int x,string &s){_insert(rt[pre],rt[x],s,s.length()-1);}
int query(int l,int r,string s){
int nowl=rt[l],nowr=rt[r],c,len=s.length();
int ret=0;
for (int i=len-1;i>=0&&(nowl||nowr);--i){
c=(s[i]-'a'+lastans)%C;
nowl=ch[nowl][c];
nowr=ch[nowr][c];
}
return cnt[nowr]-cnt[nowl];
}
}/*}}}*/
void print(string s){
int len=s.length();
for (int i=0;i<len;++i) printf("%c",s[i]);
printf("\n");
}
void get_s(string &s){
scanf("%s",tmps);
int len=strlen(tmps);
s.resize(0);
for (int i=0;i<len;++i)
s.push_back(tmps[i]);
} int main(){
#ifndef ONLINE_JUDGE
freopen("a.in","r",stdin);
#endif
char ch;
Pr rec;
scanf("%d\n",&n);
T1::init(); T2::init();
lastans=0;
for (int i=1;i<=n;++i){
get_s(s[i]);
//print(s[i]);
T1::insert(s[i],i);
}
T1::get_order();
for (int i=1;i<=lis[0];++i){
T2::insert(i-1,i,s[lis[i]]);
}
scanf("%d\n",&m);
for (int i=1;i<=m;++i){
get_s(s1);
get_s(s2);
//print(s1);print(s2);
rec=T1::query(s1);
if (rec.first!=-1)
lastans=T2::query(rec.first-1,rec.second,s2);
else lastans=0;
printf("%d\n",lastans);
}
}

【bzoj4212】神牛的养成计划的更多相关文章

  1. [BZOJ4212]神牛的养成计划

    [BZOJ4212]神牛的养成计划 试题描述 Hzwer 成功培育出神牛细胞,可最终培育出的生物体却让他大失所望...... 后来,他从某同校女神 牛处知道,原来他培育的细胞发生了基因突变,原先决定神 ...

  2. BZOJ4212 神牛的养成计划 (字典树,bitset)

    题面 Description Hzwer成功培育出神牛细胞,可最终培育出的生物体却让他大失所望- 后来,他从某同校女神 牛处知道,原来他培育的细胞发生了基因突变,原先决定神牛特征的基因序列都被破坏了, ...

  3. 【BZOJ4212】神牛的养成计划 Trie树+可持久化Trie树

    [BZOJ4212]神牛的养成计划 Description Hzwer成功培育出神牛细胞,可最终培育出的生物体却让他大失所望...... 后来,他从某同校女神 牛处知道,原来他培育的细胞发生了基因突变 ...

  4. 【BZOJ-4212】神牛的养成计划 Trie树 + 可持久化Trie树

    4212: 神牛的养成计划 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 136  Solved: 27[Submit][Status][Discus ...

  5. BZOJ 4212: 神牛的养成计划

    4212: 神牛的养成计划 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 142  Solved: 30[Submit][Status][Discus ...

  6. BZOJ.4212.神牛的养成计划(Trie 可持久化Trie)

    BZOJ 为啥hzw的题也是权限题啊 考虑能够匹配\(s1\)这一前缀的串有哪些性质.对所有串排序,能发现可以匹配\(s1\)的是一段区间,可以建一棵\(Trie\)求出来,设为\([l,r]\). ...

  7. BZOJ 4212: 神牛的养成计划 可持久化trie+trie

    思路倒是不难,但是这题卡常啊 ~ code: #include <bits/stdc++.h> #define N 2000004 #define M 1000005 #define SI ...

  8. Web前端开发工程师养成计划【转载】

    Web前端开发工程师养成计划(入门篇) 最原始的忠告:这个世界上有想法的人很多,但是有想法又能实现它的人太少! 首先要感谢伟大的Web2.0概念.产品概念.用户体验概念.jQuery插件,是它们在中国 ...

  9. 高手养成计划基础篇-Linux第二季

    高手养成计划基础篇-Linux第二季   本文来源:i春秋社区-分享你的技术,为安全加点温度   前言 前面我们学习了文件处理命令和文件搜索命令,简单的了解了一下Linux,但是仅仅了解这样还不行,遇 ...

随机推荐

  1. Zabbix部署-LNMP环境

    原文发表于cu:2016-05-05 参考文档: LNMP安装:http://www.osyunwei.com/archives/7891.html 一.环境 Server:CentOS-7-x86_ ...

  2. leetcode27_C++Remove Element

    给定一个数组 nums 和一个值 val,你需要原地移除所有数值等于 val 的元素,返回移除后数组的新长度. 不要使用额外的数组空间,你必须在原地修改输入数组并在使用 O(1) 额外空间的条件下完成 ...

  3. Hibernate入门篇<1>hibernate.cfg.xml学习小结

    Hibernate配置文件主要用于配置数据库连接和Hibernate运行时所需的各种属性,这个配置文件应该位于应用程序或Web程序的类文件夹 classes中.Hibernate配置文件支持两种形式, ...

  4. CSS布局之圣杯布局和双飞翼布局

    其实圣杯布局和双飞翼布局实现的都是三栏布局,两边的盒子宽度固定,中间盒子自适应,也就是我们常说的固比固布局.它们实现的效果都是一样的,差别在于其实现的思想. 一.圣杯布局 html代码中,将重要的内容 ...

  5. RIGHT-BICEP单元测试——“二柱子四则运算升级版”

    RIGHT-BICEP单元测试 ——“二柱子四则运算升级版” ”单元测试“这对于我们来说是一个全新的专业含义,在上了软件工程这门课,并当堂编写了简单的"求一组数中的最大值"函数的单 ...

  6. 2016-2017 ACM-ICPC, NEERC, Moscow Subregional Contest Problem L. Lazy Coordinator

    题目来源:http://codeforces.com/group/aUVPeyEnI2/contest/229511 时间限制:1s 空间限制:512MB 题目大意: 给定一个n 随后跟着2n行输入 ...

  7. 18软工实践-第八次作业(课堂实战)-项目UML设计(团队)

    目录 团队信息 分工选择 课上分工 课下分工 ToDolist alpha版本要做的事情 燃尽图 UML 用例图 状态图 活动图 类图 部署图 实例图 对象图 时序图 包图 通信图 贡献分评定 课上贡 ...

  8. Firefox必备的24款web开发插件

    from: 软件过滤: 排序:收录时间 | 浏览数 网页开发FireFox插件 Firebug Firebug是Firefox下的一款开发类插件,现属于Firefox的 五星级强力推荐插件之一.它集H ...

  9. week1读构建之法-读书笔记

    最开始听见杨老师说邹欣老师这个名字总觉得很熟悉,后来看见博客上老师的头像恍然大悟,原来机缘巧合已经在微博上关注邹老师许久,一直觉得邹老师是个很有意思的人,兴趣一定十分广泛,看了老师的书确实能感觉到邹老 ...

  10. 修改mac的hosts文件

    第一步:请先打开 Mac 系统中的 Finder 应用,接下来请按快捷键组合 Shift+Command+G 三个组合按键,并输入 Hosts 文件的所在路径:/etc/hosts , 随后即可在 F ...