失配树

名字看起来挺高级的,然而其实就是 \(\text{KMP}\) 上树啦。

我们将每个点的 \(nex[i]\)\(i\) 连边,那么最终 \(border\) 关系会形成一棵树,之后就可以在树上搞事情啦!

P5829 【模板】失配树

这题比较裸,直接根据定义建树之后对于两个前缀求出在 \(fail\) 树上的最近公共祖先即可。

$\texttt{code}$
// Author:A weak man named EricQian
#include<bits/stdc++.h>
using namespace std;
#define infll 0x7f7f7f7f7f7f7f7f
#define inf 0x3f3f3f3f
#define Maxn 1000005
#define Maxpown 22
typedef long long ll;
inline int rd()
{int x=0;char ch,t=0;while(!isdigit(ch = getchar())) t|=ch=='-';while(isdigit(ch)) x=x*10+(ch^48),ch=getchar();return x=t?-x:x;
}
int n,q,tot;
int pre[Maxn],dep[Maxn];
int fa[Maxn][Maxpown];
char s[Maxn];
inline int query(int x,int y)
{if(dep[x]>dep[y]) swap(x,y);for(int i=20;i>=0;i--) if(dep[fa[y][i]]>=dep[x]) y=fa[y][i];if(x==y) return fa[x][0];for(int i=20;i>=0;i--) if(fa[x][i]!=fa[y][i]) x=fa[x][i],y=fa[y][i];return fa[x][0];
}
int main()
{//ios::sync_with_stdio(false); cin.tie(0);//freopen(".in","r",stdin);//freopen(".out","w",stdout);scanf("%s",s+1),n=strlen(s+1);for(int i=2,j=0;i<=n;i++){while(j && s[i]!=s[j+1]) j=pre[j];if(s[i]==s[j+1]) j++;pre[i]=j,fa[i][0]=j,dep[i]=dep[j]+1;}for(int i=1;i<=20;i++)for(int j=1;j<=n;j++)fa[j][i]=fa[fa[j][i-1]][i-1];q=rd();for(int i=1,x,y;i<=q;i++) x=rd(),y=rd(),printf("%d\n",query(x,y));//fclose(stdin);//fclose(stdout);return 0;
}

小插曲:(小声)

ZCETHAN 告诉 EricQian 这一题用失配树做,EricQian 立刻表示它不会失配树。【过了 5 秒】EricQian 大声喊道:我发明了失配树!

CF432D Prefixes and Suffixes

题意:给你一个长度为 \(n\) 的长字符串,“完美子串”既是它的前缀也是它的后缀,求“完美子串”的个数且统计这些子串的在长字符串中出现的次数。

我们发现对于一个前缀的出现个数其实就是:(难以表述直接用伪代码列出了)

len=这个前缀的长度 
for(int i=1;i<=n;i++)for(int j=i;j;j=nex[j])ans+=(j==len)?1:0;

之后我们发现对于同一个前缀它可能被访问多次,这个可以直接倒换循环顺序实现 \(O(n)\) 解决。(于是你就发明的失配树)

$\texttt{code}$
// Author:A weak man named EricQian
#include<bits/stdc++.h>
using namespace std;
#define infll 0x7f7f7f7f7f7f7f7f
#define inf 0x3f3f3f3f
#define Maxn 100005
typedef long long ll;
inline int rd()
{int x=0;char ch,t=0;while(!isdigit(ch = getchar())) t|=ch=='-';while(isdigit(ch)) x=x*10+(ch^48),ch=getchar();return x=t?-x:x;
}
inline ll maxll(ll x,ll y){ return x>y?x:y; }
inline ll minll(ll x,ll y){ return x<y?x:y; }
inline ll absll(ll x){ return x>0ll?x:-x; }
inline ll gcd(ll x,ll y){ return (y==0)?x:gcd(y,x%y); }
int n,tot;
int nex[Maxn],cnt[Maxn];
int ans[Maxn][2];
char s[Maxn];
int main()
{//ios::sync_with_stdio(false); cin.tie(0);//freopen(".in","r",stdin);//freopen(".out","w",stdout);scanf("%s",s+1),n=strlen(s+1);for(int i=2,j=0;i<=n;i++){while(j && s[i]!=s[j+1]) j=nex[j];if(s[i]==s[j+1]) j++;nex[i]=j;}for(int i=n;i>=1;i--) cnt[i]++,cnt[nex[i]]+=cnt[i];
//	 for(int i=1;i<=n;i++)
//	 	 for(int j=i;j;j=nex[j]) cnt[j]++;ans[++tot][0]=n,ans[tot][1]=1;for(int i=nex[n];i;i=nex[i])ans[++tot][0]=i,ans[tot][1]=cnt[i];printf("%d\n",tot);for(int i=tot;i>=1;i--) printf("%d %d\n",ans[i][0],ans[i][1]);//fclose(stdin);//fclose(stdout);return 0;
}

P2375 [NOI2014] 动物园

这道题可以根本不用往失配树去理解,我们发现每一次最长合法 \(\text{border}\) 的长度最多只会增加 \(1\),那么直接暴力跑 \(\text{KMP}\) 并及时处理 \(\text{border}\) 长度大于一半的情况即可。

一些正确性证明:如果这个位置的 \(\text{border}\) 长度大于了 \(\frac{i}{2}\),这个 \(\text{border}\) 的后缀起始点将永远不再会成为答案,因此跳出这个 \(\text{border}\) 一定是对的。

$\texttt{code}$
// Author:A weak man named EricQian
#include<bits/stdc++.h>
using namespace std;
#define infll 0x7f7f7f7f7f7f7f7f
#define inf 0x3f3f3f3f
#define Maxn 1000005
#define mod 1000000007
typedef long long ll;
inline int rd()
{int x=0;char ch,t=0;while(!isdigit(ch = getchar())) t|=ch=='-';while(isdigit(ch)) x=x*10+(ch^48),ch=getchar();return x=t?-x:x;
}
inline ll maxll(ll x,ll y){ return x>y?x:y; }
inline ll minll(ll x,ll y){ return x<y?x:y; }
inline ll absll(ll x){ return x>0ll?x:-x; }
inline ll gcd(ll x,ll y){ return (y==0)?x:gcd(y,x%y); }
int n,ans;
char s[Maxn];
int nex[Maxn],dep[Maxn];
int main()
{//ios::sync_with_stdio(false); cin.tie(0);//freopen(".in","r",stdin);//freopen(".out","w",stdout);int T=rd();while(T--){scanf("%s",s+1),n=strlen(s+1),ans=1;dep[1]=1;for(int i=2,j=0;i<=n;i++){while(j && s[i]!=s[j+1]) j=nex[j];if(s[i]==s[j+1]) j++;nex[i]=j,dep[i]=dep[j]+1;}for(int i=2,j=0;i<=n;i++){while(j && s[i]!=s[j+1]) j=nex[j];if(s[i]==s[j+1]) j++;while(j*2>i) j=nex[j];ans=1ll*ans*(1ll*dep[j]+1ll)%mod;}printf("%d\n",ans);}//fclose(stdin);//fclose(stdout);return 0;
}

P3426 [POI2005]SZA-Template

考虑设 \(dp(i)\) 表示主串的前缀 \([1,i]\) 需要的最短长度。

容易发现在失配树上 \(dp(i)\) 沿着树成单调递增关系,考虑 \(dp(i)\)\(dp(nex_i)\) 的关系。

如果 \(dp(i)\in (dp(nex_i),nex_i]\),由于 \([1,i]\) 的一个 border 为 \(dp(nex_i)\),所以 \(dp(nex_i)\) 也一定能够生成这个串。

所以 \(dp(i)\) 能够生成的串一定是 \(dp(nex_i)\) 能够生成的真子集,即答案取 \(dp(nex_i)\) 一定比 \((dp(nex_i),nex_i]\) 优。

因此我们只用判断能否由 \(dp(nex_i)\) 生成 \([1,i]\) 即可,具体即记录 \(dp(nex_i)\) 能够生成的最长串 \(s\),如果 \(|s|+|dp(nex_i)|\ge i\) 即可。

如果上述情况不成立,意味着 \(dp(i)=i\)

$\texttt{code}$
// Author:A weak man named EricQian
#include<bits/stdc++.h>
using namespace std;
#define infll 0x7f7f7f7f7f7f7f7f
#define inf 0x3f3f3f3f
#define Maxn 500005
typedef long long ll;
inline int rd()
{int x=0;char ch,t=0;while(!isdigit(ch = getchar())) t|=ch=='-';while(isdigit(ch)) x=x*10+(ch^48),ch=getchar();return x=t?-x:x;
}
inline ll maxll(ll x,ll y){ return x>y?x:y; }
inline ll minll(ll x,ll y){ return x<y?x:y; }
inline ll absll(ll x){ return x>0ll?x:-x; }
inline ll gcd(ll x,ll y){ return (y==0)?x:gcd(y,x%y); }
int n;
int nex[Maxn],maxx[Maxn],dp[Maxn];
char s[Maxn];
int main()
{//ios::sync_with_stdio(false); cin.tie(0);//freopen(".in","r",stdin);//freopen(".out","w",stdout);scanf("%s",s+1),n=strlen(s+1);for(int i=2,j=0;i<=n;i++){while(j && s[i]!=s[j+1]) j=nex[j];if(s[i]==s[j+1]) j++;nex[i]=j;}for(int i=1;i<=n;i++){if(maxx[dp[nex[i]]]+dp[nex[i]]>=i) dp[i]=dp[nex[i]];else dp[i]=i;maxx[dp[i]]=i;}printf("%d\n",dp[n]);//fclose(stdin);//fclose(stdout);return 0;
}

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

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

相关文章

P7116-[NOIP2020]微信步数【数学】

正题 题目链接:https://www.luogu.com.cn/problem/P7116 题目大意 有一个kkk维空间&#xff0c;第iii维长度为wiw_iwi​&#xff0c;有nnn步每一步都是让某个维的坐标1/−11/-11/−1&#xff0c;每次走完nnn步都会从111重新走一次&#xff0c;现在求从这个空间的每个点出发走…

[2019CSP-S Day1]提高组Day1题解(格雷码[模拟(k转二进制取反的做法带证明)] + 括号树[DP] + 树上的数(暴力+菊花图+单链))

Day1T1&#xff1a;格雷码题目题解代码实现T2&#xff1a;括号树题目题解代码实现T3&#xff1a;树上的数题目10pts暴力题解代码实现25pts菊花图题解代码实现25pts单链题解代码实现T1&#xff1a;格雷码 题目 通常&#xff0c;人们习惯将所有 n位二进制串按照字典序排列&…

模板:环套树

文章目录前言解析找环代码练习环套树的直径代码thanks for reading&#xff01;前言 环套树者&#xff0c;一个环套一棵树也 解析 定义&#xff1a;n个点&#xff0c;n条边的无向连通图 其实就是树多了一条边&#xff0c;连出了一个环 性质&#xff1a;如果对环套树进行dfs&am…

使用PerfView监测.NET程序性能(四):折叠,过滤和时间范围选择

在上一篇文章使用PerfView监测.NET程序性能&#xff08;三&#xff09;&#xff1a;分组中&#xff0c;我们使用了Perfview的分组功能。分组功能旨在对某些函数按照某个格式进行分组&#xff0c;以减少视图中的各种无关函数的数量。但仅有分组还不够&#xff0c;有时我们想将一…

P1352 没有上司的舞会

P1352 没有上司的舞会 题意&#xff1a; 给你一个树&#xff0c;每个点都有权值&#xff0c;选择一些点使得权值和最大&#xff0c;要求父亲节点和子节点不能同时选择 题解&#xff1a; 经典树形dp dp[x][0]表示以x为根的子树&#xff0c;且x不参加舞会的最大快乐值 dp[x][…

基础字符串

初三巨佬djwj233 基础字符串

CF536C-Tavas and Pashmaks【凸壳】

正题 题目链接:https://codeforces.com/contest/536/problem/C 题目大意 nnn个人&#xff0c;第iii个人的游泳速度sis_isi​&#xff0c;跑步速度是rir_iri​。如果跑道长度是RRR&#xff0c;泳道长度是SSS那么一个人的用时就是RriSsi\frac{R}{r_i}\frac{S}{s_i}ri​R​si​S​…

带旋treap概念及模板,带例题:普通平衡树

带旋Treap二叉查找树BST(Binary Search Tree)定义Treap定义模板合集&#xff08;均为O(logn)O(logn)O(logn)&#xff09;push_up模板旋转模板插入模板删除模板查找前驱模板查找后驱模板查找键值key模板查找节点的修正值rank模板PS&#xff1a;rd的比较问题例题&#xff1a;普通…

微服务系列实践 .NET CORE

从事这个行业转眼已经6年了&#xff0c;从当初刚毕业的在北京朝八晚十&#xff0c;从二环到五环&#xff0c;仍每天精力充沛的小愤青&#xff1b;再到深圳一点一滴的辛勤在软件行业的耕种&#xff0c;从当初单体应用架构到现在微服务架构的经历&#xff0c;回想起来自己的收获倒…

P2607 [ZJOI2008]骑士

P2607 [ZJOI2008]骑士 题意&#xff1a; n个点n个边&#xff0c;每个点都有权值&#xff0c;相邻的点不能同时选择&#xff0c;问如何选择能使得权值最大 题解&#xff1a; 这个题很有P1352 没有上司的舞会这个题的感觉&#xff0c;唯一的区别是那个题保证是树&#xff0c;…

模板:线段树优化建图

前言 百川到海&#xff0c;天下归一 解析 线段树优化建图是用于对一个区间的点连边时的优化方法 建一棵in树一棵出树分别往上和下指即可 大概长这样 &#xff08;pia的洛谷的照片&#xff09; 建树 正常动态开点即可 void build(int &k,int l,int r){tr[ktot](tree){0…

Binary Search

01 分数划分 什么 01 分数划分&#xff0c;叫 Binary Search 多好。 P1642 规划 可以二分答案 \(x\)&#xff0c;考虑选择 \(n-m\) 个数使得答案 \(\ge x\)。 \[\dfrac{\sum w(i)}{\sum c(i)}\ge x \]\[\sum w(i)\ge \sum (x\times c(i)) \]\[\sum(w(i)-x\times c(i))\ge 0 \]之…

CF1556F-Sports Betting【状压dp,数学期望】

正题 题目链接:https://www.luogu.com.cn/problem/CF1556F 题目大意 nnn个点的一张竞赛图&#xff0c;每个点有一个权值aia_iai​&#xff0c;(i,j)(i,j)(i,j)之间的边iii连jjj的概率是aiaiaj\frac{a_i}{a_ia_j}ai​aj​ai​​&#xff0c;否则jjj连iii。 现在期望有多少个点…

[非旋平衡树]fhq_treap概念及模板,例题:普通平衡树,文艺线段树

文章目录概念全套模板push_up模板split拆树模板(按权值拆)split拆树模板(按个数拆)merge合并模板&#xff08;地址版&#xff09;merge合并模板&#xff08;带返回根&#xff09;区间模板insert插入模板delete删除模板find_kth找第k大模板get_rank找排名模板pre找前驱模板suf找…

P4381 [IOI2008]Island

P4381 [IOI2008]Island 题意&#xff1a; 给你一棵基环树森林&#xff0c;求出基环树的直径之和. 题解&#xff1a; 对于基环树,我们将环看作根,那么直径有两种情况:: 1.不经过环,也就是环上某个点的子树内部,对于这种情况,直接在子树内部处理直径,更新答案即可; 2.经过环…

20210912模拟

文章目录总结复盘总结 190pts 50403070 今天吸取了昨天的教训&#xff0c;没有再像之前一样先死磕一道题&#xff0c;把命运押在切题上&#xff0c;整体时间分配还算合理 要认真审题&#xff01; 今天T1少看了条件&#xff0c;T4一开始看错题意浪费了时间… 复盘 先看题 乍…

surging 微服务引擎 1.0 正式发布

surging 是一个分布式微服务引擎,提供高性能RPC远程服务调用&#xff0c;服务引擎支持http、TCP、WS、Mqtt协议,采用Zookeeper、Consul作为surging服务的注册中心&#xff0c;集成了哈希一致性&#xff0c;随机&#xff0c;轮询、压力最小优先作为负载均衡的算法&#xff0c;底…

CF1594F-Ideal Farm【构造】

正题 题目链接:https://www.luogu.com.cn/problem/CF1594F 题目大意 给出n,s,kn,s,kn,s,k&#xff0c;求是否所有的长度为nnn且和为sss的正整数序列都有一段和为kkk的区间。 1≤T≤105,1≤n,s,k≤10181\leq T\leq 10^5,1\leq n,s,k\leq 10^{18}1≤T≤105,1≤n,s,k≤1018 解题…

P4178 Tree

P4178 Tree 题意&#xff1a; 给定一棵 n 个节点的树&#xff0c;每条边有边权&#xff0c;求出树上两点距离小于等于 k 的点对数量。 题解&#xff1a; 点分治的模板题是求等于K的路径条数 本题是求小于等于K的路径条数&#xff0c;我们只需要改变统计答案即可 原本统计答…

YBTOJ:彩色圆环

文章目录前言题目描述InputOutputSample InputSample Output解析代码前言 尽信书&#xff0c;则不如无书 题目描述 Input 仅有一行&#xff0c;该行给出依次两个正整数N, M&#xff0c;分别表示宝石的个数和宝石在变化时可能变成的颜色种类数。 Output 应仅有一行&#xff0…