加强版[BZOJ#3483] SGU505 Prefixes and suffixes(询问在线版)

文章目录

  • description
  • solution
  • code

#3483. SGU505 Prefixes and suffixes(询问在线版)

description

给定𝑛个字符串,有𝑚个询问。
每个询问给出两个字符串𝑠1, 𝑠2,问𝑛个字符串中有多少个字符串满足𝑠1是其前缀,且𝑠2是其后缀。

输入格式
从文件 fix.in 中读入数据。
输入文件的第一行包含一个整数𝑛,含义如上所述。
接下来𝑛行,每行一个字符串。
接下来一行包含一个整数𝑚,代表询问数。
接下来𝑚行,每行包含两个字符串𝑠1, 𝑠2,代表一次询问。询问内容请参见问题描述。
为了体现算法的在线性,我们令上次询问的答案是𝑥(如果这是第一个询问则𝑥 = 0)。询问的两个字符串的所有字符都要按字母表的顺序向后移动𝑥位(字母表是一个环)。例如 x=2,输入的 s1=qz 则查询的 s1=sb。

输出格式
输出到 fix.out 中。
对于每次询问,输出串满足既是𝑠1的前缀,又是𝑠2的后缀的字符串的数量。

样例输入

10
emikuqihgokuhsywlmqemihhpgijkxdukjfmlqlwrpzgwrwozkmlixyxniutssasrriafu
emikuqihgokuookbqaaoyiorpfdetaeduogebnolonaoehthfaypbeiutssasrriafu
emikuqihgokuorocifwwymkcyqevdtglszfzgycbgnpomvlzppwrigowekufjwiiaxniutssasrriafu
emikuqihgokuorociysgfkzpgnotajcfjctjqgjeeiheqrepbpakmlixyxniutssasrriafu
emikuqihgokuorociysgfrhulymdxsqirjrfbngwszuyibuixyxniutssasrriafu
emikuqihgokuorguowwiozcgjetmyokqdrqxzigohiutssasrriafu
emikuqihgokuorociysgsczejjmlbwhandxqwknutzgdmxtiutssasrriafu
emikuqihgokuorociysgvzfcdxdiwdztolopdnboxfvqzfzxtpecxcbrklvtyxniutssasrriafu
emikuqihgokuorocsbtlyuosppxuzkjafbhsayenxsdmkmlixyxniutssasrriafu
emikuqihgokuorociysgfjvaikktsixmhaasbvnsvmkntgmoygfxypktjxjdkliixyxniutssasrriafu
10
emikuqihgokuorociysg yxniutssasrriafu
aiegqmedckgqknky eqpoowonnewbq
xfbdnjbazhdnhkhvb qrqgbnmlltlkkbtyn
bjfhrnfedlhrlolzfv qppxpoofxcr
zhdfpldcbjf stsidponnvnmmdvap
zhdfpldcbjfpjmjxdt gdstsidponnvnmmdvap
dlhjtphgfnjtnqnbhxr wxwmhtsrrzrqqhzet
bjfhrnfedlhrlolzfv frqppxpoofxcr
zhdfpldcbjf dponnvnmmdvap
ucyakgyxweakehes nondykjiiqihhyqvk

样例输出

4
7
3
5
5
1
3
5
10
4

数据规模和约定
设𝐿1表示𝑛个字符串的长度之和,𝐿2表示𝑚次查询的字符串𝑠1, 𝑠2的长度之和。

数据点𝑛的规模𝑚的规模L1的规模L2的规模
1= 10= 10≤ 1000≤ 500
2= 200= 1000≤ 160000≤ 160000
3= 500= 5000≤ 400000≤ 800000
4= 1000= 10000≤ 800000≤ 400000
5~7= 2000= 50000≤ 1600000≤ 1600000
8~10= 2000= 100000≤ 1600000≤ 3200000

对于所有数据,保证字符串的字符集为小写英文字母

solution

刚开始就想了前缀字典树和后缀字典树,但是又要求是同一个字符串,如果暴力存储每个字典树节点代表的子串被包含在哪些字符串里面,似乎就可以前缀后缀对比求出

但这显然会TLETLETLE,可能跟暴力差不多

害怕空间超限制,我又想到了可持久化字典树,但似乎不大有作用


考试过程中,游老师突然说了句

排完序,就是线性复杂度了

waitaminute!\rm wait\ a\ minute!wait a minute!

我好像没有排序诶——我陷入了沉思

排序??

排什么序?排序为我带来了什么好处吗??

排序会让序列变得有序~~(废话)~~

树的dfn序,使得树可以被划分成连续区间

..................

真実はいつも一つ!

当我将字符串排序后,对于一段前缀,包含它的字符串一定是一段连续区间!!!

于是乎可以像树的dfn序求节点管辖的子树区间,我可以求出包含该前缀的字符串区间的左右端点

答案通过这一步就进行了字符串筛选,留下的这个区间恰好是所有满足s1s1s1是其前缀的字符串

接下来就是进一步筛选,这些字符串中后缀是s2s2s2的串

首先我怎么做到只要这一段区间里面的字符串信息呢??

暴力枚举——这又回到了起点,不可取

是的没错,就是一开始的——可持久化

这样就转化成了前缀和问题,rrr版本减去l−1l-1l1版本即可

s2s2s2为后缀,似乎也是一开始的——后缀字典树

对,没错,倒着再来一个可持久化字典树

时间复杂度主要是在前缀字典树和后缀字典树上跑s1,s2s1,s2s1,s2长度的字符串求和

前面的一些预处理,复杂度没有超过查询的复杂度

两个部分无交集,复杂度是相加关系

所以取大复杂度而言,的确是线性的


只要想到了,就非常好写这个代码

正解其实就是把之前的乱想法,安排了一个顺序,前提是拍好了序而已


以上是正解做法

接下来再分享一种方法把

字符串一般都容易想到——哈希

尤其是想不出正解的时候,哈希总归是不会抛弃你的

这道题就是预处理对于每一个字符串的每一个前缀/后缀都哈希一下

一共是2∗L12*L12L1

然后查询的时候,就直接枚举nnn个字符串,然后直接调第iii个字符串长度为s1,s2s1,s2s1,s2的前缀、后缀哈希,跟查询的两个串哈希比较即可

这种做法是O(nm)O(nm)O(nm)

code

#include <cstdio>
#include <iostream>
#include <algorithm>
using namespace std;
#define maxn 2002
#define maxm 1600002
int n, m, len, cnt_pre, cnt_suf;
string s[maxn];
int L[maxm], R[maxm], sum[maxm], root[maxn];
int pre[maxm][26], suf[maxm][26];void insert_pre( int id ) {len = s[id].length();int now = 0;for( int i = 0;i < len;i ++ ) {int k = s[id][i] - 'a';if( ! pre[now][k] ) pre[now][k] = ++ cnt_pre;now = pre[now][k];if( ! L[now] ) L[now] = id;R[now] = id;}
}void insert_suf( int lst, int &now, int id, int d ) {if( ! ~d ) return;now = ++ cnt_suf;for( int i = 0;i < 26;i ++ ) suf[now][i] = suf[lst][i];sum[now] = sum[lst] + 1;int k = s[id][d] - 'a';insert_suf( suf[lst][k], suf[now][k], id, d - 1 );
}string a, b;int find() {int now = 0;for( int i = 0;i < len;i ++ ) {int k = a[i] - 'a';if( ! pre[now][k] ) return 0;else now = pre[now][k];}return now;
}int query( int l, int r, int d ) {int k = b[d] - 'a';if( sum[suf[r][k]] - sum[suf[l][k]] == 0 ) return 0;if( d == 0 ) return sum[suf[r][k]] - sum[suf[l][k]];return query( suf[l][k], suf[r][k], d - 1 );
}int main() {freopen( "fix.in", "r", stdin );freopen( "fix.out", "w", stdout );scanf( "%d", &n );for( int i = 1;i <= n;i ++ ) cin >> s[i];sort( s + 1, s + n + 1 );for( int i = 1;i <= n;i ++ ) insert_pre( i ), insert_suf( root[i - 1], root[i], i, len - 1 );scanf( "%d", &m );int ans = 0;while( m -- ) {cin >> a >> b;len = a.length();for( int i = 0;i < len;i ++ ) {int x = a[i] - 'a';x = ( x + ans ) % 26;a[i] = x + 'a';}int pos = find();if( ! pos ) {printf( "%d\n", ans = 0 );continue;}len = b.length();for( int i = 0;i < len;i ++ ) {int x = b[i] - 'a';x = ( x + ans ) % 26;b[i] = x + 'a';}int l = L[pos], r = R[pos];printf( "%d\n", ans = query( root[l - 1], root[r], len - 1 ) );}return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/316997.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

AcWing 320. 能量项链

AcWing 320. 能量项链 题意&#xff1a; 题解&#xff1a; 和环形石头合并基本一样 代码&#xff1a; #include<bits/stdc.h> #define debug(a,b) printf("%s %d\n",a,b); typedef long long ll; using namespace std;inline int read(){int s0,w1;char c…

ERP不规范,同事两行泪

最近的很多次对外交流&#xff0c;都聊到了ERP建设的话题&#xff0c;并且无一例外的不那么让人省心&#xff0c;回想我这么多年走过的ERP坑坑路&#xff0c;在这里也写下经验和总结&#xff0c;希望能给正在或者即将走上ERP建设路的企业一些思考和帮助。导读1、几个瞎眼而普遍…

2.13模拟总结

文章目录前言题目解析最小划分&#xff08;divide&#xff09;进制路径&#xff08;base&#xff09;欧拉欧拉&#xff08;eular&#xff09;代码T1T2T3总结前言 day9 170pts 期望&#xff1a;10010020220 实际&#xff1a;701000170 rnk7 挂的分有点多qwq 分数要是得满就能拿…

Panasonic Programming Contest (AtCoder Beginner Contest 195) 题解

文章目录A - Health M DeathB - Many OrangesC - CommaD - Shipping CenterE - Lucky 7 BattleF - Coprime PresentPanasonic Programming Contest (AtCoder Beginner Contest 195)A - Health M Death 判断倍数。 #include <cstdio> int main() {int M, H;scanf( "…

长沙.NET社区之光

奈何万事开头难迎着改革开放四十年带来的春风&#xff0c;长沙的互联网生态环境以唐胡子俱乐部为首的一众互联网社群将长沙互联网的环境推上了一个新的台阶。年底&#xff0c;我与有幸一起共事的溪源兄&#xff0c;下班后一起闲聊&#xff0c;觉着长沙的.NET的生态环境亟待改善…

P3205 [HNOI2010]合唱队

P3205 [HNOI2010]合唱队 题意&#xff1a; 有n个数&#xff0c;然后插入队伍中&#xff0c;如果队列当前为空&#xff0c;则直接插入&#xff0c;然后每次插入和上一次插入的比较&#xff0c;如果大于&#xff0c;插入当前队列的最右侧&#xff0c;如果小于&#xff0c;插入当…

2.14模拟总结

前言 节日快乐&#xff01; (逃) day10 50pts 期望&#xff1a;10302060 实际&#xff1a;0302050 rnk11 彻彻底底的摆烂局了。 但是rnk竟然没有太掉&#xff0c;所以我应该并不孤独… 和KH并排坐在机房里&#xff0c;各自看着电脑&#xff0c;痴痴想着各自的心事&#xff0c;…

KYOCERA Programming Contest 2021(AtCoder Beginner Contest 200)题解

文章目录A - CenturyB - 200th ABC-200C - Ringos Favorite Numbers 2D - Happy Birthday! 2E - Patisserie ABC 2F - Minflip SummationKYOCERA Programming Contest 2021&#xff08;AtCoder Beginner Contest 200&#xff09; A - Century 简单的除以200200200向上取整 B…

高级进阶:Azure DevOps搞定.NET Core编译版本号自增

点击上方蓝字关注“汪宇杰博客”熟悉.NET Framework的人知道&#xff0c;我们可以通过指定AssemblyVersion为10.0.*来让编译器自增版本号。但是.NET Core和.NET Standard不行。即使有MSBump这样的开源项目&#xff0c;也有一定的缺陷。一般这样的需求会出现在CI/CD服务器上。我…

Little Boxes UVALive - 8209

Little Boxes UVALive - 8209 题意&#xff1a; 给你四个数&#xff0c;输出四个数之和&#xff0c;四个数小于等于262之内 题解&#xff1a; 这。。。这。。水题 unsigned int 0&#xff5e;4294967295 (10位数&#xff0c;4e9) int -2147483648&#xff5e;2147483647 (…

2.15模拟总结

前言 day11 期望&#xff1a;406030130 实际&#xff1a;4003070 rnk16 挂大分了。。 T2树边不加双向&#xff1a;60->0。 这什么伞兵bug啊&#xff01; 整体状态也不太好&#xff0c;T2死磕无果。 T1看出正解结果写不出来拉插&#xff0c;乐。 题目解析 T1 网格序列&am…

Mynavi Programming Contest 2021(AtCoder Beginner Contest 201)题解

文章目录A - Tiny Arithmetic SequenceB - Do you know the second highest mountain?C - Secret NumberD - Game in Momotetsu WorldE - Xor DistancesF - Insertion SortMynavi Programming Contest 2021&#xff08;AtCoder Beginner Contest 201&#xff09;A - Tiny Arit…

Newbe.Claptrap - 一套以 “事件溯源” 和“Actor 模式”作为基本理论的服务端开发框架...

本文是关于 Newbe.Claptrap 项目主体内容的介绍&#xff0c;读者可以通过这篇文章&#xff0c;大体了解项目内容。轮子源于需求随着互联网应用的蓬勃发展&#xff0c;相关的技术理论和实现手段也在被不断创造出来。诸如 “云原生架构”、“微服务架构”、“DevOps” 等一系列关…

Rabbits UVALive - 8211

Rabbits UVALive - 8211 题意&#xff1a; n个兔子的位置&#xff0c;兔子每次可以跳到两个兔子之间&#xff0c;问最多可以跳多少下&#xff1f; 题解&#xff1a; 求出所有相邻两数的间隔&#xff0c;然后减去最小间隔就是答案 代码&#xff1a; #include <bits/std…

2.16模拟总结

前言 期望&#xff1a;100700170 实际&#xff1a;400040 rnk14 分全部挂没了&#xff0c;太行了。 T1不开longlong见祖宗&#xff0c;而且KH说的那个也有道理&#xff0c;带权之后树的重心可以不只有两个&#xff0c;所以最后还应该倍增的跳。&#xff08;然而这个地方题解似…

Caddi Programming Contest 2021(AtCoder Beginner Contest 193) 题解

Caddi Programming Contest 2021(AtCoder Beginner Contest 193) A - Discount 打折浮点数除即可 B - Play Snuke 枚举判断符合要求的求最小值即可 C - Unexpressed O(n)O(\sqrt{n})O(n​)枚举aaa&#xff0c;暴力翻倍&#xff08;最小的222最多乘323232次就会超过nnn的上…

NET Core微服务之路:SkyWalking+SkyApm-dotnet分布式链路追踪系统的分享

对于普通系统或者服务来说&#xff0c;一般通过打日志来进行埋点&#xff0c;然后再通过elk或splunk进行定位及分析问题&#xff0c;更有甚者直接远程服务器&#xff0c;直接操作查看日志&#xff0c;那么&#xff0c;随着业务越来越复杂&#xff0c;企业应用也进入了分布式服务…

YBTOJBZOJ:大根堆(启发式合并)

解析 如果到了链上&#xff0c;本题就是求LIS。 考虑LIS的常见队列做法&#xff0c;其本质就是维护长度为 xxx 的序列的结尾的最小值 qxq_xqx​。 那么在本题尝试如法炮制&#xff0c;对于每个节点&#xff0c;都开一个队列&#xff0c;qxq_xqx​ 表示子树内选了 xxx 个节点后…

Tree UVALive - 8212

Tree UVALive - 8212 题意&#xff1a; 有n个点&#xff0c;k个颜色&#xff0c;每个点都要被染色&#xff0c;相同颜色之间的边算是被该颜色覆盖&#xff0c;问有多少边被所有颜色覆盖 题解&#xff1a; 题目给的是无根树&#xff0c;我们可以将1默认为根然后求所有点的子…

dotnetcore-officeaddin-toolbox : Office 365 Add-in开发人员的工具箱

在上一篇文章&#xff08;.NET Core开源行动&#xff1a;一键创建Excel Add-in&#xff09; 中我给大家展示了一套为Office 365 Add-in开发人员准备的模板库&#xff0c;你可以通过 dotnet new excel & dotnet run 命令即可完成一个新的Add-in的创建和运行。关于如何加载这…