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

Time Limit: 20 Sec  Memory Limit: 128 MB
Submit: 649  Solved: 305
[Submit][Status]

Description

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

Input

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

Output

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

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

2
1
0
1 2
【提示】
事实上样例给出的数据如果翻译成地球上的语言可以这样来看
2 3
izayoi sakuya
orihara izaya
izay
hara
raiz

HINT

【数据范围】

对于30%的数据,保证:

1<=N,M<=1000,喵星人的名字总长不超过4000,点名串的总长不超过2000。

对于100%的数据,保证:

1<=N<=20000,1<=M<=50000,喵星人的名字总长和点名串的总长分别不超过100000,保证喵星人的字符串中作为字符存在的数不超过10000。

Source

很裸很裸的AC自动机。只不过这道题让我基本搞懂了stl map的用法。

map.insert(pair) 插入映射关系

map.find(x) 查询x映射的关系,返回一个迭代器,可用first,second查找

map<int,int>::iterator it1; 迭代器

另外AC自动机的一个小细节,让我Wa了两遍:

fail指针如果需要指向深度为1的节点时要特判,而且对于已经确定fail指针的节点,切忌重新复制(其实就是if语句条件要严密)。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<map>
#include<vector>
using namespace std;
#define MAXT 201000
#define MAXN MAXT
vector<int> query[MAXN];
int query_gets[MAXN];
int cat_gets[MAXN];
int qur_vis[MAXN];
int topq=-;
struct aaa
{
int id;
int fail;
int qur_id;
map<int,int> next;
aaa()
{
qur_id=id=fail=-;
}
};
int root;
aaa tree[MAXT];
int topt=;
int name[MAXN],topn=-;
int str[MAXN];
int n,m; void add_str(int * str,int qur)
{
int ind;
int now=root;
map<int,int>::iterator it1;
while (*str!=-)
{
ind=*(str++);
it1=tree[now].next.find(ind);
if (it1==tree[now].next.end())
{
tree[now].next.insert(make_pair(ind,++topt));
tree[topt].id=ind;
now=topt;
}else
{
now=it1->second;
}
}
if (tree[now].qur_id==-)
{
tree[now].qur_id=++topq;
query[topq].push_back(qur);
}else
{
query[tree[now].qur_id].push_back(qur);
}
}
int q[MAXT];
void build_tree()
{
int ope=,clo=,now,temp,x,ind;
map<int,int>::iterator it1,it2;
tree[].fail=;
for (it1=tree[].next.begin();it1!=tree[].next.end();it1++)
{
now=it1->second;
q[++clo]=now;
tree[now].fail=;
}
while (ope<clo)
{
now=q[++ope];
for (it1=tree[now].next.begin();it1!=tree[now].next.end();it1++)
{
x=it1->second;
ind=it1->first;
q[++clo]=x;
temp=tree[now].fail;
while (temp!=root)
{
it2=tree[temp].next.find(ind);
if (it2!=tree[temp].next.end())
{
tree[x].fail=it2->second;
break;
}
temp=tree[temp].fail;
}
it2=tree[root].next.find(ind);
if (tree[x].fail==-&&it2!=tree[root].next.end())//Oh, I forgot this part again
{
tree[x].fail=it2->second;
}
if (tree[x].fail==-)
{
tree[x].fail=;
}
}
}
} int count_str(int *str,int cat_id)
{
int ret=;
int now=;
int ind,x,temp;
int i;
map<int,int>::iterator it1;
while (*str!=-)
{
ind=*(str++);
it1=tree[now].next.find(ind);
while (now!=root&&it1==tree[now].next.end())
{
now=tree[now].fail;
it1=tree[now].next.find(ind);
}
if (it1==tree[now].next.end())
{
now=root;
}else
{
now=it1->second;
}
temp=now;
while (temp!=root)
{
if (tree[temp].qur_id!=-)
{
x=tree[temp].qur_id;
for (i=;i<query[x].size();i++)
{
if(qur_vis[query[x][i]]!=cat_id)
{
query_gets[query[x][i]]++;
ret++;
qur_vis[query[x][i]]=cat_id;
}
}
}
temp=tree[temp].fail;
} }
return ret;
}
void init()
{
topt=;
tree[].id=-;
root=;
int i,j,x,y;
scanf("%d%d",&n,&m);
for (i=;i<n*;i++)
{
scanf("%d",&x);
name[++topn]=x;
for (j=;j<x;j++)
{
scanf("%d",&y);
name[++topn]=y;
}
name[++topn]=-;
}
for (i=;i<m;i++)
{
scanf("%d",&x);
for (j=;j<x;j++)
{
scanf("%d",&y);
str[j]=y;
}
str[x]=-;
add_str(str,i);
}
build_tree();
memset(qur_vis,-,sizeof(qur_vis));
}
void work()
{
int i;
int now=;
for (i=;i<n*;i++)
{
now++;
cat_gets[i/]+=count_str(&name[now],i/);
now+=name[now-];
now++;
}
for (i=;i<m;i++)
{
printf("%d\n",query_gets[i]);
}
for (i=;i<n-;i++)
{
printf("%d ",cat_gets[i]);
}
printf("%d\n",cat_gets[n-]);
}
int main()
{
init();
work();
return ;
}

BZOJ 2754: [SCOI2012]喵星球上的点名的更多相关文章

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

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

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

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

  3. 【刷题】BZOJ 2754 [SCOI2012]喵星球上的点名

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

  4. bzoj 2754 [SCOI2012]喵星球上的点名(后缀数组)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2754 [题意] 每只喵有名姓,如果被老师点到名或姓的子串都要答道,但每只喵一次点名只答 ...

  5. BZOJ 2754 [SCOI2012]喵星球上的点名 (AC自动机、树状数组)

    吐槽: 为啥很多人用AC自动机暴力跳都过了?复杂度真的对么? 做法一: AC自动机+树状数组 姓名的问题,中间加个特殊字符连起来即可. 肯定是对点名串建AC自动机(map存儿子),然后第一问就相当于问 ...

  6. BZOJ 2754 [SCOI2012]喵星球上的点名 (AC自动机+map维护Trie树)

    题目大意:略 由于字符集大,要用map维护Trie树 并不能用AC自动机的Trie图优化,不然内存会炸 所以我用AC自动机暴跳fail水过的 显然根据喵星人建AC自动机是不行的,所以要根据问题建 然而 ...

  7. bzoj 2754: [SCOI2012]喵星球上的点名【AC自动机】

    洛谷90,最后一个点死活卡不过去(也可能是我写的有问题? 比较暴力的做法,把询问带着标号建立AC自动机,用map存儿子. 然后用名字串在自动机上跑,以为是名或姓的子串就行所以把名和姓中间加个特殊字符拼 ...

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

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

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

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

随机推荐

  1. NuGet的使用心得

    前言 上星期发布了NuGet的使用和服务搭建后,同时NuGet在部门里也使用了起来.经过这些天的使用,总结了些小技巧和注意点,希望和大家分享下. 问题提出 使用了NuGet的朋友们估计都知道,在签入代 ...

  2. [D3] 14. Line and Area Charts with D3

    <!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8&quo ...

  3. linux中的帮助命令 分类: linux 学习笔记 ubuntu 2015-07-05 19:07 31人阅读 评论(0) 收藏

    说实话,到目前为止我还是不太习惯使用linux自带的帮助文档,遇到问题都是去查我自己下载的chm格式的命令大全,不过这些帮助命令我们还是有必要了解的. 1.man [要查看的命令名称] 例如想要查看l ...

  4. System Operations on AWS - Lab 1W - Creating EC2 (Windows)

    1. 创建CommandHost实例,登录到CommandHost,通过AWS CLI创建WebServer实例. 1.1 为CommandHost实例创建一个IAM角色 1.2 创建CommandH ...

  5. RedHat7配置IdM server

    IdM服务器是一个集成身份验证服务器. Figure 1.1. The IdM Server: Unifying Services Authentication: Kerberos KDC Kerbe ...

  6. build/envsetup.sh内lunch解析

    ........ # 测试device是否存在且是一个目录 并且 只查找device目录4层以上的子目录,名字为vendorsetup.sh 并且 将命令执行的错误报告直接送往回收站 不显示在屏幕上 ...

  7. Codeforces Round #310 (Div. 2)--B

    http://codeforces.com/problemset/problem/556/B 题意:给定n个数字且都小于n,然后每次循环第2k+1个数字+1,第2k个数字减一,k=0,1,2...n/ ...

  8. js给当前日期加一天

    <script type="text/javascript"> function addDay(datetime, days) { var old_time = new ...

  9. winfrom面向对象1

    1:面向对象的技术概论 要学习好面向对象,我们应该从三个问题入手: 1.什么是面向对象? 2.为什么要面向对象? 3.该怎么面向对象? 对象的定义是人们要进行研究的任何事物,从最简单的整数到复杂的飞机 ...

  10. (正则表达式应用) 替换自闭合标签(self-closing tag)的method

    var str = "<sup><div class=\"he's\"/></sup><span id=\"cs\&q ...