http://codevs.cn/problem/2403/

2012年省队选拔赛四川

 时间限制: 2 s
 空间限制: 128000 KB
 
题目描述 Description

a180285幸运地被选做了地球到喵星球的留学生。他发现喵星人在上课前的点名现象非常有趣。 
假设课堂上有 N 个喵星人,每个喵星人的名字由姓和名构成。喵星球上的老师会选择M 个串来点名,每次读出一个串的时候,如果这个串是一个喵星人的姓或名的子串,那么这个喵星人就必须答到。 
然而,由于喵星人的字码过于古怪,以至于不能用 ASCII 码来表示。为了方便描述,a180285决定用数串来表示喵星人的名字。 
现在你能帮助 a180285 统计每次点名的时候有多少喵星人答到,以及 M 次点名结束后每个喵星人答到多少次吗?

 
输入描述 Input Description

现在定义喵星球上的字符串给定方法: 
先给出一个正整数 L,表示字符串的长度,接下来L个整数表示字符串的每个字符。 
输入的第一行是两个整数N和M。 
接下来有N行,每行包含第 i个喵星人的姓和名两个串。姓和名都是标准的喵星球上的字符串。 
接下来有M行,每行包含一个喵星球上的字符串,表示老师点名的串。

 
输出描述 Output Description

对于每个老师点名的串输出有多少个喵星人应该答到。 
然后在最后一行输出每个喵星人被点到多少次。

 
样例输入 Sample Input

2 3 
6 8 25 0 24 14 8 6 18 0 10 20 24 0 
7 14 17 8 7 0 17 0 5 8 25 0 24 0 
4 8 25 0 24 
4 7 0 17 0 
4 17 0 8 25

 
样例输出 Sample Output




1 2

 
数据范围及提示 Data Size & Hint

对于30%的数据,保证: 
1 <= M, N <= 1000
喵星人的名字总长不超过 4000,点名串的总长不超过 2000, 
对于100%的数据,保证: 
1<=N<=20000
1<=M<=50000
喵星人的名字总长和点名串的总长分别不超过
100000 保证喵星人的字符串中作为字符存在的数不超过 10000

AC自动机

学了几招:

1、用map搞trie

2、bool数组初始化,数组很大,需清空的数目不多,用vector记下来,枚举vector里的元素,清零

(貌似CDQ分治、整体二分就是这样搞的)

#include<iostream>
#include<cstdio>
#include<queue>
#include<map>
#include<vector>
#define N 100010
using namespace std;
map<int,int>trie[N];
vector<int>a[],st[N],USE,MARK;
int f[N];
int len,tot=,n,m;
queue<int>q;
bool mark[N],use[];
int call[],miao[];
struct aa
{
void insert(int i)
{
int x,root=;
while(len--)
{
scanf("%d",&x);
if(!trie[root][x]) trie[root][x]=++tot;
root=trie[root][x];
}
st[root].push_back(i);
}
void getfail()
{
q.push();int j;
while(!q.empty())
{
int now=q.front();q.pop();
map<int,int>::iterator it;
for(it=trie[now].begin();it!=trie[now].end();it++)
{
int w=it->second,e=it->first;
q.push(w);
j=f[now];
while(!trie[j][e]) j=f[j];
f[w]=trie[j][e];
}
}
}
void get(int now,int who)
{
for(int i=now;i;i=f[i])
{
if(mark[i]) return;
else
{
mark[i]=true;
MARK.push_back(i);
for(int j=;j<st[i].size();j++)
{
if(!use[st[i][j]])
{
use[st[i][j]]=true;
USE.push_back(st[i][j]);
miao[who]++;
call[st[i][j]]++;
}
}
}
}
}
void solve(int k)
{
int id,root=;
for(int i=;i<a[k].size();i++)
{
id=a[k][i];
while(!trie[root][id]) root=f[root];
root=trie[root][id];
get(root,k);
}
for(int i=;i<MARK.size();i++) mark[MARK[i]]=false;
for(int i=;i<USE.size();i++) use[USE[i]]=false;
MARK.clear();USE.clear();
}
}tree;
int main()
{
/*freopen("name.in","r",stdin);
freopen("name.out","w",stdout);*/
int x;
scanf("%d%d",&n,&m);
for(int i=;i<=n;i++)
{
scanf("%d",&len);
while(len--)
{
scanf("%d",&x);
a[i].push_back(x);
}
a[i].push_back(-);
scanf("%d",&len);
while(len--)
{
scanf("%d",&x);
a[i].push_back(x);
}
}
for(int i=;i<=m;i++)
{
scanf("%d",&len);
tree.insert(i);
}
for(int i=-;i<=;i++) trie[][i]=;
tree.getfail();
for(int i=;i<=n;i++)
tree.solve(i);
for(int i=;i<=m;i++) printf("%d\n",call[i]);
for(int i=;i<n;i++) printf("%d ",miao[i]);printf("%d",miao[n]);
return ;
}

SCOI2012喵星球上的点名的更多相关文章

  1. BZOJ 2754: [SCOI2012]喵星球上的点名

    2754: [SCOI2012]喵星球上的点名 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 649  Solved: 305[Submit][Sta ...

  2. BZOJ2754: [SCOI2012]喵星球上的点名

    2754: [SCOI2012]喵星球上的点名 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 680  Solved: 314[Submit][Sta ...

  3. BZOJ 2754: [SCOI2012]喵星球上的点名 [后缀数组+暴力]

    2754: [SCOI2012]喵星球上的点名 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 1906  Solved: 839[Submit][St ...

  4. BZOJ 2754: [SCOI2012]喵星球上的点名 [AC自动机+map+暴力]

    2754: [SCOI2012]喵星球上的点名 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 1902  Solved: 837[Submit][St ...

  5. P2336 [SCOI2012]喵星球上的点名(后缀自动机+莫队+dfs序)

    P2336 [SCOI2012]喵星球上的点名 名字怎么存?显然是后缀自动机辣 询问点到多少个喵喵喵其实就是 查询后缀自动机上parent树的一个子树 于是我们考虑莫队 怎么树上莫队呢 我们用dfs序 ...

  6. 洛咕 P2336 [SCOI2012]喵星球上的点名

    洛咕 P2336 [SCOI2012]喵星球上的点名 先求出SA和height,一个点名串对应的就是一段区间,还有很多个点,就转化成了 有很多个区间,很多个点集,对每个区间计算和多少个点集有交,对每个 ...

  7. 洛谷 P2336 [SCOI2012]喵星球上的点名 解题报告

    P2336 [SCOI2012]喵星球上的点名 题目描述 a180285 幸运地被选做了地球到喵星球的留学生.他发现喵星人在上课前的点名现象非常有趣. 假设课堂上有 \(N\) 个喵星人,每个喵星人的 ...

  8. 【BZOJ2754】[SCOI2012]喵星球上的点名

    [BZOJ2754][SCOI2012]喵星球上的点名 题面 bzoj 洛谷 题解 这题有各种神仙做法啊,什么暴力\(AC\)自动机.\(SAM\)等等五花八门 我这个蒟蒻在这里提供一种复杂度正确且常 ...

  9. 2754. [SCOI2012]喵星球上的点名【后缀数组】

    Description a180285幸运地被选做了地球到喵星球的留学生.他发现喵星人在上课前的点名现象非常有趣.   假设课堂上有N个喵星人,每个喵星人的名字由姓和名构成.喵星球上的老师会选择M个串 ...

随机推荐

  1. Beta 冲刺 (6/7)

    队名:Boy Next Door 燃尽图 代码写入 https://github.com/mangoqiqi/paybook/tree/master/Desktop/Web%E8%B4%A6%E5%8 ...

  2. JAVA自学日记——Part Ⅰ.

    和C++比较相似,Java同样是面向对象的设计语言,在基础的语句上有一些不大的差别,经过两天的学习,大概的了解了在eclipse中如何进行简单的编程,解决一些简单的问题,诸如在学习C时做过的“字符串倒 ...

  3. 材料设计---Design

    效果: main_activity.xml <?xml version="1.0" encoding="utf-8"?> <!--Coordi ...

  4. 简单复利计算java板

    一.要求: 1.客户说:帮我开发一个复利计算软件. 2如果按照单利计算,本息又是多少呢? 3.假如30年之后要筹措到300万元的养老金,平均的年回报率是3%,那么,现在必须投入的本金是多少呢? 4.利 ...

  5. 从解决一个java.lang.NoSuchMethodError想到的

    今天在发布系统部署一个web app的时候,发现应用服务器(tomcat 7.0.26)不能正常启动,于是远程登陆到服务器上查看应用服务器的启动日志,在tomcat_home的logs/localho ...

  6. uva 1513(线段树)

    题目链接:1513 - Movie collection 题意:有一堆电影,按1-n顺序排,有m次操作,每次询问第ai个电影之前有多少个电影,然后将其抽出放在堆顶. 分析:线段树应用. 因为每次查询后 ...

  7. c++ 替换修改一个文件夹下的所有文件的文件名

    代码简洁,亲测可用. 1,首先来获取(输出)一个文件夹中所有的文件名 void getFiles(string path, vector<string>& files) { //文 ...

  8. BZOJ 2580: [Usaco2012 Jan]Video Game

    2580: [Usaco2012 Jan]Video Game Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 142  Solved: 96[Subm ...

  9. 洛谷 P1072 Hankson 的趣味题 解题报告

    P1072 \(Hankson\)的趣味题 题目大意:已知有\(n\)组\(a0,a1,b0,b1\),求满足\((x,a0)=a1\),\([x,b0]=b1\)的\(x\)的个数. 数据范围:\( ...

  10. 如何在 Android 程序中禁止屏幕旋转和重启Activity

    禁止屏幕随手机旋转变化 有时候我们希望让一个程序的界面始终保持在一个方向,不随手机方向旋转而变化:在AndroidManifest.xml的每一个需要禁止转向的Activity配置中加入android ...