【BZOJ4896】[Thu Summer Camp2016]补退选 Trie树

【BZOJ4896】[Thu Summer Camp2016]补退选

Description

X是T大的一名老师,每年他都要教授许多学生基础的C++知识。在T大,每个学生在每学期的开学前都需要选课,每

次选课一共分为三个阶段:预选,正选,补退选;其中"补退选"阶段最忙碌。在补退选阶段,学生即可以选课,也

可以退课。对于X老师来说,在补退选阶段可能发生以下两种事件:

1:一个姓名为S的学生选了他的课(姓名S将出现在X的已选课学生名单中)

2:一个姓名为S的学生退了他的课(姓名S将从X的已选课学生名单中移除)

同时,X老师对于有哪些学生选了他的课非常关心,所以他会不定时的查询已选课学生名单,每次查询的格式如下

:最早在哪个事件之后,姓名以S为前缀的学生数量超过了vX老师看你骨骼惊奇,所以想用这个问题考考你,你当

然不会畏惧,所以勇敢的接下了这个任务。

注意1:学生的姓名可能相同,如果有p个姓名相同的学生都选了X老师的课,则他们的姓名将出现在X老师的名单上p次。

注意2:只有已经选了课的学生才会退课,如果姓名为S的学生退课,则在他退课之前X老师的名单上一定有姓名S。

注意3:选课,退课和查询都被定义为"事件","事件"的编号从1开始

Input

第一行包含一个正整数n,表示一共发生了n个事件。

接下来n行,每行描述一个事件;每行第一个正整数k表示事件类型:

1.如果k=1,表示选课事件,接下来一个字符串S,表示一个姓名为S的学生选了X老师的课

2.如果k=2,表示退课事件,接下来一个字符串S,表示一个姓名为S的学生退了X老师的课

3.如果k=3,表示查询事件,接下来一个字符串S以及三个非负整数a,b,c,表示X老师想知道最早在第几个事件之后

,姓名以S为前缀的学生数量超过了(a*|ANS|+b)%c,|ANS|表示上次查询事件的答案的绝对值,如果当前是第一次

查询,则|ANS|=0;如果任何时刻都没有超过该值,则答案为-1。注:输入中的所有字符串均只包含小写字母。

Output

对于每个查询事件,输出一行表示该查询答案。

Sample Input

6
1 lcaq
3 lca 0 0 5
1 lcq
3 lc 0 1 5
2 lcaq
3 lc 0 1 5

Sample Output

1
3
3

HINT

n<=100000,字符串长度 <= 60,输入中的所有字符串只会包含前 10 个小写字母

题解:直接给Trie树上的每个节点都开一个vector,记录它第一次人数超过i的时间,就没有然后了~

只包含前10个小写字母是骗人的,不要信

#include <cstdio>
#include <cstring>
#include <iostream>
#include <vector>
using namespace std;
struct node
{
	int ch[26],siz;
	vector<int> v;
}p[300000];
int n,tot,ans;
char str[70];
int main()
{
	int i,j,a,b,c,d,l,u;
	scanf("%d",&n);
	tot=1;
	for(i=1;i<=n;i++)
	{
		scanf("%d%s",&d,str),l=strlen(str);
		if(d==1)
		{
			for(u=1,j=0;j<l;j++)
			{
				a=str[j]-‘a‘;
				if(!p[u].ch[a])	p[u].ch[a]=++tot;
				u=p[u].ch[a],p[u].siz++;
				if(p[u].siz>p[u].v.size())	p[u].v.push_back(i);
			}
		}
		if(d==2)
		{
			for(u=1,j=0;j<l;j++)	a=str[j]-‘a‘,u=p[u].ch[a],p[u].siz--;
		}
		if(d==3)
		{
			for(u=1,j=0;j<l&&u;j++)	a=str[j]-‘a‘,u=p[u].ch[a];
			scanf("%d%d%d",&a,&b,&c),a=((long long)a*ans+b)%c;
			if(!u||a>=p[u].v.size())
			{
				ans=1;
				printf("-1\n");
				continue;
			}
			ans=p[u].v[a];
			printf("%d\n",ans);
		}
	}
	return 0;
}
时间: 2024-08-23 22:47:07

【BZOJ4896】[Thu Summer Camp2016]补退选 Trie树的相关文章

BZOJ 4896 :[Thu Summer Camp2016]补退选 Trie树+Vector

4896: [Thu Summer Camp2016]补退选 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 315  Solved: 97[Submit][Status][Discuss] Description X是T大的一名老师,每年他都要教授许多学生基础的C++知识.在T大,每个学生在每学期的开学前都需要选课,每 次选课一共分为三个阶段:预选,正选,补退选:其中"补退选"阶段最忙碌.在补退选阶段,学生即可以选课,也 可以退课.对于

BZOJ4896 [Thu Summer Camp2016]补退选

本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000 作者博客:http://www.cnblogs.com/ljh2000-jump/转载请注明出处,侵权必究,保留最终解释权! 题目链接:BZOJ4896 正解:$Trie+vector$ 解题报告: $THUSC2016$ $T2$ 考虑每次加入删除对出现次数的影响只会加减一,那么我可以先用$Trie$来维护整个问题中出现的字符串,然后对于每个节点我都记录一下以当前串为前缀的字符

[THUSC2016]补退选 - Trie

Description 维护一个字符串集合,有三种事件,加入一个字符串,删除一个字符串,询问最早在哪个事件之后,以某个串为前缀的字符串数量超过\(k\),强制在线.\(n \le 100000,|S| \le 60\),输入中的所有字符串只会包含前\(10\)个小写字母. Solution 建立\(Trie\)树维护字符串集合,在\(Trie\)树上的每一个节点开一个\(vector\),用来记录子树内的结束节点到达某个值的最早时刻,查询时直接找到前缀对应节点即可. Code #include

【BZOJ4103】[Thu Summer Camp 2015]异或运算 可持久化Trie树

[BZOJ4103][Thu Summer Camp 2015]异或运算 Description 给定长度为n的数列X={x1,x2,...,xn}和长度为m的数列Y={y1,y2,...,ym},令矩阵A中第i行第j列的值Aij=xi xor  yj,每次询问给定矩形区域i∈[u,d],j∈[l,r],找出第k大的Aij. Input 第一行包含两个正整数n,m,分别表示两个数列的长度 第二行包含n个非负整数xi 第三行包含m个非负整数yj 第四行包含一个正整数p,表示询问次数 随后p行,每行

bzoj4896 补退选

Description X是T大的一名老师,每年他都要教授许多学生基础的C++知识.在T大,每个学生在每学期的开学前都需要选课,每 次选课一共分为三个阶段:预选,正选,补退选:其中"补退选"阶段最忙碌.在补退选阶段,学生即可以选课,也 可以退课.对于X老师来说,在补退选阶段可能发生以下两种事件: 1:一个姓名为S的学生选了他的课(姓名S将出现在X的已选课学生名单中) 2:一个姓名为S的学生退了他的课(姓名S将从X的已选课学生名单中移除) 同时,X老师对于有哪些学生选了他的课非常关心,所

T2 选ID trie树

T2 选ID Description ? 机房似乎有许多人没有 ID,热心同志小 A 为了服务人民,所有决定帮大家找一个合适的 ID. ? 小 A 觉得一个合适的 ID,和这个人的相关程度应该是比较高的,就像他的 ID 里有他的名字缩写一样.为了量化这个相关程度,他定义一个人的名字 \(S\) 和他的 ID \(T\) 的匹配值的大小为\(lcp\)(\(S\),\(T\))?.现在有 \(n\) 个人,小 A 想好了 \(n\) 个 ID,他想知道如果把 ID 分配给机房众人,最后能够得到的最

4.15 省选模拟赛 编码 trie树 前缀和优化建图 2-sat

好题 np. 对于20分 显然可以爆搜. 对于50分 可以发现每个字符串上的问号要么是0,要么是1.考虑枚举一个字符串当前是0还是1 这会和其他字符串产生矛盾. 所以容易 发现这是一个2-sat问题. 拆点 把任意两个产生矛盾的字符串进行连边.然后最后判矛盾即可. n^2枚举 建图 判断矛盾时使用字符串hash 要分类讨论4种情况. using namespace std; const int MAXN=1010,maxn=500010,cc1=19260817,cc2=114514; int

usaco6.1-Cow XOR:trie树

Cow XOR Adrian Vladu -- 2005 Farmer John is stuck with another problem while feeding his cows. All of his N (1 ≤ N ≤ 100,000) cows (numbered 1..N) are lined up in front of the barn, sorted by their rank in their social hierarchy. Cow #1 has the highe

BZOJ 3012 [Usaco2012 Dec]First! wzq脑洞hash树(正解trie树)

博客风格转化计划实验篇2 题意: 给n(n<=30000)个字符串,所有字符串长度加起来不超过300000(字符串只含有小写字母). 求解哪些字符串可以称作第一字符串. 一个字符串能被称为第一字符串的条件为存在一种字典序使它排在第一名. 方法: wzq脑洞hash树-.. 然而并没有trie树优越. 并且我脑洞的这个树好处有啥:暂且不知道. 坏处有啥:很容易被卡,自带常数. 所以为什么要这么写?只是因为我跟wjc说这题我特么一定要用带hash的东西搞过去! *目标达成√ 解析: 对于以下内容,请