BZOJ 2754 [SCOI2012]喵星球上的点名 (AC自动机、树状数组)
吐槽: 为啥很多人用AC自动机暴力跳都过了?复杂度真的对么?
做法一: AC自动机+树状数组
姓名的问题,中间加个特殊字符连起来即可。
肯定是对点名串建AC自动机(map存儿子),然后第一问就相当于问每个姓名串(以下称作“关键路径”)经过了多少个点名串(以下称做“关键点”)在fail树中的子树中的至少一点,第二问就相当于问你每条关键路径被多少个关键点经过了在fail树的子树中至少一个点。
所以对于每个关键路径在AC自动机上跑,每跑到一个点把它到根的路径上打上标记(注意每个姓名串要有不同的标记),最后统计标记个数即可。
然后如果暴力跳去更新可过,我不知道复杂度对不对,感觉是错的。(也许是\(O(n\sqrt n)\)?)
脑子里第一思路是用bitset, 可以在暴力的复杂度基础上除以\(\omega\), 没试过
第二思路是线段树/启发式合并,没仔细想
最后看了一波题解: 我们的目标就是让同一关键路径上的点只被加一次,这样就可以变或为加,不需要状压bitset了。
然后一种好办法是像虚树那样按DFS序排序,相邻两个求出LCA,在LCA到根的路径上-1. 差分树状数组维护即可。
时间复杂度\(O(n\log n)\)
易错点: 注意AC自动机不补成Trie图,绝对不能再fail[son[u][i]]=son[fail[u]][i]了!(一晚上的惨痛教训……) 即使是写板子也要经过脑子!!!!!!!
听说也可以SA+主席树?并不会233
**UPD: **BZOJ2780是本题的一部分,求每个点名串是多少姓名串的子串,SA的做法是把所有点名串连成一起然后求SA,转化成了求区间内有多少种不同的颜色。也有更简单的SAM做法,不用任何数据结构\(O(n)\), 对所有姓名串建广义SAM,再求出来每个点有多少个姓名串经过,然后拿点名串在上面跑,跑到结尾节点输出个数即可。
代码
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<vector>
#include<map>
#include<algorithm>
using namespace std;
const int N = 5e4;
const int S = 5e4+1;
const int SIZ = 1e5;
const int LGSIZ = 16;
struct Edge
{
int v,nxt;
} e[(SIZ<<1)+3];
int fe[SIZ+3];
vector<int> a[N+3];
vector<int> b[N+3];
vector<int> ky;
map<int,int> son[SIZ+3];
int fail[SIZ+3];
int id[N+3];
int tr[SIZ+3];
int que[SIZ+3];
int ansa[N+3],ansb[N+3];
int dfn[SIZ+3];
int fa[SIZ+3][LGSIZ+3];
int sz[SIZ+3];
int dep[SIZ+3];
int num[SIZ+3];
int n,m,siz,en,cnt;
void addval(int lrb,int val)
{
while(lrb<=cnt)
{
tr[lrb] += val;
lrb += (lrb&(-lrb));
}
}
int querysum(int rb)
{
int ret = 0;
while(rb>0)
{
ret += tr[rb];
rb -= (rb&(-rb));
}
return ret;
}
int insertstr(vector<int> str)
{
int u = 0;
for(int i=0; i<str.size(); i++)
{
if(son[u].count(str[i])==0) {siz++; son[u][str[i]] = siz;}
u = son[u][str[i]];
}
num[u]++;
return u;
}
void buildACA()
{
int head = 1,tail = 0;
for(map<int,int>::iterator iter=son[0].begin(); iter!=son[0].end(); iter++)
{
int u = (iter->second);
tail++; que[tail] = u; fail[u] = 0;
}
while(head<=tail)
{
int u = que[head]; head++;
for(map<int,int>::iterator iter=son[u].begin(); iter!=son[u].end(); iter++)
{
int v = (iter->second),i = (iter->first);
if(v)
{
int uu = fail[u];
while(uu && !son[uu].count(i)) {uu = fail[uu];}
fail[v] = son[uu][i];
tail++; que[tail] = v;
}
}
}
}
void addedge(int u,int v)
{
en++; e[en].v = v;
e[en].nxt = fe[u]; fe[u] = en;
}
void dfs(int u)
{
cnt++; dfn[u] = cnt;
sz[u] = 1;
for(int i=1; i<=LGSIZ; i++) fa[u][i] = fa[fa[u][i-1]][i-1];
for(int i=fe[u]; i; i=e[i].nxt)
{
if(e[i].v==fa[u][0]) continue;
fa[e[i].v][0] = u;
dep[e[i].v] = dep[u]+1;
num[e[i].v] += num[u];
dfs(e[i].v);
sz[u] += sz[e[i].v];
}
}
int LCA(int u,int v)
{
if(dep[u]<dep[v]) swap(u,v);
int dif = dep[u]-dep[v];
for(int i=LGSIZ; i>=0; i--) {if(dif&(1<<i)) {u = fa[u][i];}}
if(u==v) return u;
for(int i=LGSIZ; i>=0; i--) {if(fa[u][i]!=fa[v][i]) {u = fa[u][i]; v = fa[v][i];}}
return fa[u][0];
}
bool cmp_dfn(int x,int y) {return dfn[x]<dfn[y];}
int main()
{
scanf("%d%d",&n,&m); siz = 0;
for(int i=1; i<=n; i++)
{
int len; scanf("%d",&len); for(int j=1; j<=len; j++) {int x; scanf("%d",&x); a[i].push_back(x);}
a[i].push_back(S);
scanf("%d",&len); for(int j=1; j<=len; j++) {int x; scanf("%d",&x); a[i].push_back(x);}
}
for(int i=1; i<=m; i++)
{
int len; scanf("%d",&len); for(int j=1; j<=len; j++) {int x; scanf("%d",&x); b[i].push_back(x);}
id[i] = insertstr(b[i]);
}
buildACA();
for(int i=1; i<=siz; i++)
{
addedge(fail[i],i); addedge(i,fail[i]);
}
cnt = 0; dfs(0);
for(int i=1; i<=n; i++)
{
int u = 0; ky.push_back(u);
for(int j=0; j<a[i].size(); j++)
{
while(u && !son[u][a[i][j]]) {u = fail[u];}
u = son[u][a[i][j]]; if(u) ky.push_back(u);
}
sort(ky.begin(),ky.end(),cmp_dfn);
for(int j=0; j<ky.size(); j++)
{
if(j>0)
{
int lca = LCA(ky[j],ky[j-1]);
addval(dfn[lca],-1); ansa[i] -= num[lca];
}
addval(dfn[ky[j]],1); ansa[i] += num[ky[j]];
}
ky.clear();
}
for(int i=1; i<=m; i++)
{
ansb[i] = querysum(dfn[id[i]]+sz[id[i]]-1)-querysum(dfn[id[i]]-1);
}
for(int i=1; i<=m; i++) printf("%d\n",ansb[i]);
for(int i=1; i<=n; i++) printf("%d ",ansa[i]);
return 0;
}
BZOJ 2754 [SCOI2012]喵星球上的点名 (AC自动机、树状数组)的更多相关文章
- BZOJ 2754: [SCOI2012]喵星球上的点名 [AC自动机+map+暴力]
2754: [SCOI2012]喵星球上的点名 Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 1902 Solved: 837[Submit][St ...
- BZOJ 2754 [SCOI2012]喵星球上的点名 (AC自动机+map维护Trie树)
题目大意:略 由于字符集大,要用map维护Trie树 并不能用AC自动机的Trie图优化,不然内存会炸 所以我用AC自动机暴跳fail水过的 显然根据喵星人建AC自动机是不行的,所以要根据问题建 然而 ...
- BZOJ 2754: [SCOI2012]喵星球上的点名
2754: [SCOI2012]喵星球上的点名 Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 649 Solved: 305[Submit][Sta ...
- BZOJ 2754: [SCOI2012]喵星球上的点名 [后缀数组+暴力]
2754: [SCOI2012]喵星球上的点名 Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 1906 Solved: 839[Submit][St ...
- 【刷题】BZOJ 2754 [SCOI2012]喵星球上的点名
Description a180285幸运地被选做了地球到喵星球的留学生.他发现喵星人在上课前的点名现象非常有趣. 假设课堂上有N个喵星人,每个喵星人的名字由姓和名构成.喵星球上的老师会选择M个串来点 ...
- BZOJ2754: [SCOI2012]喵星球上的点名(AC自动机)
Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 2816 Solved: 1246[Submit][Status][Discuss] Descript ...
- BZOJ2754: [SCOI2012]喵星球上的点名(AC自动机/后缀自动机)
Description a180285幸运地被选做了地球到喵星球的留学生.他发现喵星人在上课前的点名现象非常有趣. 假设课堂上有N个喵星人,每个喵星人的名字由姓和名构成.喵星球上的老师会选择M个串 ...
- bzoj 2754 [SCOI2012]喵星球上的点名(后缀数组)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2754 [题意] 每只喵有名姓,如果被老师点到名或姓的子串都要答道,但每只喵一次点名只答 ...
- bzoj 2754: [SCOI2012]喵星球上的点名【AC自动机】
洛谷90,最后一个点死活卡不过去(也可能是我写的有问题? 比较暴力的做法,把询问带着标号建立AC自动机,用map存儿子. 然后用名字串在自动机上跑,以为是名或姓的子串就行所以把名和姓中间加个特殊字符拼 ...
随机推荐
- charts柱状图,定时刷新
option: var dataoptionone = { title : { text: '数据存储情况', subtext: '数据表', x:'center' }, tooltip: { sho ...
- WebSocket握手总结
网址:http://blog.csdn.net/edwingu/article/details/44040961 WebSocket protocol 是HTML5一种新的协议.它实现了浏览器与服务器 ...
- [luoguP4142]洞穴遇险
https://www.zybuluo.com/ysner/note/1240792 题面 戳我 解析 这种用来拼接的奇形怪状的东西,要不就是轮廓线\(DP\),要不就是网络流. 为了表示奇数点(即\ ...
- 慕课网3-13编程练习:采用flex弹性布局制作页面主导航
小伙伴们,伸缩容器的属性我们已经学完了,接下来使用我们所学的伸缩容器属性完成下面的效果图. 要求: 1.logo.导航项.登录注册按钮这三项在水平和垂直方向上都对齐,而且他们之间的距离也相等. 2.导 ...
- 哈夫曼编码译码系统(c/c++)
哈夫曼编码译码系统的实现,主要包含三部分: 1.创建哈夫曼树 2.编码函数 3.译码函数 编写代码时为了方便,在这里混用了c++的输入输出流.主体用c语言实现. 下面时代码部分: 1.头文件,以及储存 ...
- leetCode----day02---- 买卖股票的最佳时机 II
要求: 给定一个数组,它的第 i 个元素是一支给定股票第 i 天的价格. 设计一个算法来计算你所能获取的最大利润.你可以尽可能地完成更多的交易(多次买卖一支股票). 注意:你不能同时参与多笔交易(你必 ...
- JavaScript--编程题
某班的成绩出来了,现在老师要把班级的成绩打印出来. 效果图: XXXX年XX月X日 星期X--班级总分为:81 格式要求: 1.显示打印的日期. 格式为类似“XXXX年XX月XX日 星期X” 的当前的 ...
- 关于将电脑背景+chrome等网页改成护眼豆沙绿
常用电脑的人都知道,白色等其他对比度大的颜色对眼伤害大,所以需换成柔和的豆沙绿,可长时间保证眼睛的不疲劳 windows浏览器: >>>>在桌面点右键,依次选属性(proper ...
- org.eclipse.jdt.internal.compiler.classfmt.ClassFormatException。
jdk1.8环境tomcat运行项目报错, org.eclipse.jdt.internal.compiler.classfmt.ClassFormatException.解决方法:更改jdk1.7
- Android 从服务器获取时间戳转换为年月日
用JAVA相关类转换.代码如下: Calendar calendar = Calendar.getInstance(); calendar.setTimeInMillis(NumberUtils.ge ...