CF207C3 Game with Two Trees

news/2024/9/28 15:27:27

CF207C3 Game with Two Trees

妙到家的树上字符串问题。

约定

  • \(1\)\(t_1\)

  • \(2\)\(t_2\)

  • \(S_{1/2}(i,l)\) 为树 \(1/2\) 上节点 \(i\) 沿父亲走经过 \(l\)​ 条边所构成的字符串。

  • \(E_{1/2}(u,v)\) 为树 \(1/2\) 上,连接节点 \(u,v\)​ 的边的字符。

  • \(fa_{1/2}u\) 为树 \(1/2\)\(u\) 的父亲。

前置

  • Tire 树
  • 字符串哈希
  • 树上倍增
  • 树链剖分

思路

part 1:寻找对应点

我们先把树 \(1\) 建成一棵 Tire 树,把树 \(2\) 建成一棵正常的树。

我们对于每个树 \(2\) 上的点 \(u\) 寻找一个第一颗树上的点 \(i\)\(l\)\(i\) 的深度,根深度为 \(0\)),满足:

  • \(S_1(i,l)=S_2(u,i)\)
  • \(l\) 最大。

两个条件必须都满足,由于树 \(1\) 是一棵 Tire 树,所以 \(i\) 是唯一存在的,我们把这样的 \(i\) 称为对应点。

我们可以枚举树 \(2\) 的节点 \(u\),节点 \(i\) 从树 \(1\) 的根开始,如果 \(i\) 的儿子 \(v\) 使得 \(E_2(u,fa_2u)=E_1(rt,v)\),那么我们把 \(u\) 向上走,把 \(i\) 走向 \(i\) 的儿子 \(v\)

这样可以暴力的找到我们需要的点。

但是肯定也会获得 TLE 的好成绩

我们考虑优化这个过程,可以树上的做字符串 hash 判断某一段链(从祖先到子孙)是否相等,现在要解决的问题是怎么速的爬树(寻找点)。

首先 \(u\) 是向上跳,我们可以想到倍增,但 \(i\) 需要向下,怎么解决呢?

其实 \(i\) 的主要问题在于需要判断下一条边是否有需要的边,我们可以对树 \(1\) 进行树链剖分,这样子每次肯定是向下走一条链,确定了 \(i\) 的方向。

然后枚举 \(k\)\(k\) 从大到小,且 \(2^k\) 小于 \(i\) 所在重链向下的长度)表示 \(u\) 向上爬 \(2^k\) 个点,\(i\) 向下跳 \(2^k\) 个点,然后判断跳的这一段边的字符串是否相等,相等就跳过去。

如果 \(1\) 个节点都跳不了,那么我们就手动判断 \(i\) 是否可以向下,如果可以,就向下;如果不行,则 \(i\) 点以是最优选择。

分析一下这里的复杂度,如果可以沿着重链向下跳,那么这条重链最多跳 \(\log n\) 次。跳完重链或者重链无法向下以后,跳一条轻边,而根据树剖的性质,最多跳 \(\log n\) 条轻边,也就是最多换 \(\log n\) 条重链。

总时间复杂度 \(\log^2 n\)

inline int fr(int u,int v,int s)//树 1 点 u;树 2 点 v
{for(int i=lg[min(len[v],dep[u])];~i;i--)if((h[s]-h[fa[u][i]]+mod)%mod==val[id[dfn[v]+(1<<i)]]*base[dep[fa[u][i]]]%mod)//沿重链跳u=fa[u][i],v=id[dfn[v]+(1<<i)];if(u==1||!tr[v][to[u]]) return v;//判断有没有轻边可以跳return fr(fa[u][0],tr[v][to[u]],s);//跳轻边
}

这里树 \(1\) 是 Tire 树进行了树剖,树 \(2\) 是正常树进行了倍增。

part 2 反馈询问

求得了上面的对应点,我们要怎样利用对应点去求答案呢?

part 2.1 树 2 加入点

假设这个是树 \(1\)

这个是树 \(2\)

现在树 \(2\)\(9\) 号节点的对于点是树 \(1\)\(6\) 号节点。

我们有 \(S_1(6,5)=abccc\),有 \(S_2(9,5)=cccba\)

假设其余节点已经加入,现在加入的节点是 \(9\) 号节点。

那么 \(9\) 号节点在树 \(1\) 上的答案可以是 \(1、2、3、4、5、6\) 号节点共 \(6\) 个。

不难发现树 \(1\) 上对于点所在的到根的链上的任意一个点都可以成为 \(9\) 号点的答案。

也很好证明,因为树 \(1\) 从上到下需要匹配树 \(2\) 从下到上,那么树 \(1\) 到对应点的每一个前缀,都是树 \(2\) 向上爬的后缀。

但有些时候,树 \(1\) 上的对应点可能还没有加入,所以我们对于树 \(1\) 我们需要动态的维护一条链上的点是否加入。

这里可以搭配 dfn 序食用树状数组,每次插入一个树 \(1\)​ 的节点,我们对于它子树这一个 dfn 序区间整体加 1,查询直接单点查询该点的值,是不是动态维护出了一条链上的节点。

part 2.2 树 1 加入节点

这个会稍微复杂一点点,我们还是先画个图:

这棵树是树 \(1\),圈住的红色点还没有加入。

至于边上的字母呢?这不重要

假设图中蓝色的节点是被树 \(2\) 上的节点定为对应点。

你说这种对应不可能?对不起,这也不重要

假设我们现在插入了树 \(1\) 中的 \(2\) 号节点,那么所有在 \(2\) 号节点的子树内,被树 \(2\) 对应的次数就是 \(2\) 号节点答案。

分析和 part 2.1 朴素情况下的分析一致。

每次树 \(2\) 中插入一个节点,我们就向树 \(1\) 对应点 dfn 序的位置加 \(1\)​,每次查询区间查询子树即可。

这个就是单点修改区间查询。

还是可以食用树状数组。

//T1 树 2 查询使用的;T2 是树 1 查询使用的b[1]=1;for(int i=2;i<=n2;i++) b[i]=fr(i,1,i);//求对应点T1.updata(dfn[1],1),T1.updata(dfn[1]+siz[1],-1);//区间加T2.updata(dfn[b[1]],1);//单点加ans=1;//两个 1 号节点互相匹配,答案为 1for(int i=1,x=1,y=1;i<=m;i++){if(op[i]==1)//加入树 1 点{int u=a[++x];//a 为 Tire 树上点编号ans+=T2.getsum(dfn[u]+siz[u]-1)-T2.getsum(dfn[u]-1);T1.updata(dfn[u],1),T1.updata(dfn[u]+siz[u],-1);}else//加入树 2 点{int u=b[++y];//b 为对应点编号ans+=T1.getsum(dfn[u]);T2.updata(dfn[u],1);}printf("%lld\n",ans);}

如果看不懂树状数组的话,也可以手搓线段树。

CODE

#include<bits/stdc++.h>
using namespace std;#define ll long long
#define N 3e5
#define mod 998244353const int maxn=3e5+5;inline int read() {int s = 0, w = 1;char ch = getchar();while (ch < '0' || ch > '9') {if (ch == '-')w = -1;ch = getchar();}while (ch >= '0' && ch <= '9') s = s * 10 + ch - '0', ch = getchar();return s * w;
}struct treearray
{int tree[maxn];inline int lowbit(int x){return x&(-x);}inline void updata(int x,int y){for(;x<=N;x+=lowbit(x)) tree[x]+=y;}inline int getsum(int x){int sum=0;for(;x;x-=lowbit(x)) sum+=tree[x];return sum;}
}T1,T2;int n1,n2,m;
int a[maxn],b[maxn];int idx;
int tr[maxn][27],siz[maxn],hso[maxn],dfn[maxn],id[maxn],len[maxn];
int op[maxn*2],to[maxn],fa[maxn][20],dep[maxn];int lg[maxn*2];
ll base[maxn*2],h[maxn],val[maxn];ll ans;inline void dfs1(int u)
{siz[u]=1;for(int i=0;i<26;i++){int v=tr[u][i];if(!v) continue;val[v]=(131*val[u]+i+47)%mod;dfs1(v);siz[u]+=siz[v];if(siz[v]>siz[hso[u]]) hso[u]=v;}
}
inline void dfs2(int u)
{dfn[u]=++idx;id[idx]=u;if(!hso[u]) return;dfs2(hso[u]);len[u]=len[hso[u]]+1;for(int i=0;i<26;i++){int v=tr[u][i];if(!v||v==hso[u]) continue;dfs2(v);}
}
inline int fr(int u,int v,int s)
{for(int i=lg[min(len[v],dep[u])];~i;i--)if((h[s]-h[fa[u][i]]+mod)%mod==val[id[dfn[v]+(1<<i)]]*base[dep[fa[u][i]]]%mod)u=fa[u][i],v=id[dfn[v]+(1<<i)];if(u==1||!tr[v][to[u]]) return v;return fr(fa[u][0],tr[v][to[u]],s);
}int main()
{// freopen("20.in","r",stdin);// freopen("droot.out","w",stdout);// scanf("%d",&m);m=read();n1=n2=1;dep[1]=a[1]=1;base[0]=1;for(int i=1;i<=m;i++) base[i]=base[i-1]*131%mod;for(int i=1;i<=m;i++){char c;int u;// scanf("%d%d",&op[i],&u);op[i]=read(),u=read();while(1){c=getchar();if('a'<=c&&c<='z') break;}if(op[i]==1){n1++;if(!tr[a[u]][c-'a']) tr[a[u]][c-'a']=n1;a[n1]=tr[a[u]][c-'a'];}else{n2++;dep[n2]=dep[u]+1;fa[n2][0]=u;to[n2]=c-'a';h[n2]=(h[u]+(47ll+to[n2])*base[dep[u]])%mod;}}dfs1(1);dfs2(1);lg[0]=-1;for(int i=2;i<=m;i++) lg[i]=lg[i>>1]+1;for(int j=1;j<=lg[n2];j++) for(int i=1;i<=n2;i++) fa[i][j]=fa[fa[i][j-1]][j-1];b[1]=1;for(int i=2;i<=n2;i++) b[i]=fr(i,1,i);T1.updata(dfn[1],1),T1.updata(dfn[1]+siz[1],-1);T2.updata(dfn[b[1]],1); ans=1;for(int i=1,x=1,y=1;i<=m;i++){if(op[i]==1){int u=a[++x];ans+=T2.getsum(dfn[u]+siz[u]-1)-T2.getsum(dfn[u]-1);T1.updata(dfn[u],1),T1.updata(dfn[u]+siz[u],-1);}else{int u=b[++y];ans+=T1.getsum(dfn[u]);T2.updata(dfn[u],1);}printf("%lld\n",ans);}
}

最后代码就懒得注释喽。

后记

实际上如果要做出这题,更多时候应该是先想:

  1. 要怎么求答案——》对应点
  2. 对应点要怎么求——》树剖加倍增

还是一道很有启发性的题目。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈,一经查实,立即删除!

相关文章

Python环境变量设置与读取

★ 环境变量基本概念环境变量定义环境变量是操作系统中存储有关操作系统配置信息和应用程序运行环境的动态值的一种机制。环境变量的主要作用是为正在运行的进程提供配置信息,帮助程序找到所需的资源或者确定程序运行的方式。在操作系统中,每个进程都有自己的环境变量集合。这…

使用IDEA创建编写代码

使用IDEA创建编写代码 创建空项目文件夹选择Empty Project为工程文件夹命名选择文件夹创建路径点击Create创建Java工程模块 点击File->New->Module...选择Java,为工程文件命名,选择路径,点击Create。项目结构选择 点击File->Project Structure...SDK选择自己的JDK文…

SolidState 靶机 walkthrough

扫描 ┌──(root㉿kali)-[/home/kali] └─# nmap -T5 -A -v -p- 192.168.80.141 Starting Nmap 7.92 ( https://nmap.org ) at 2022-10-24 03:50 EDT NSE: Loaded 155 scripts for scanning. NSE: Script Pre-scanning. Initiating NSE at 03:50 Completed NSE at 03:50, 0…

测试一下编辑器

这是个1级🚗标题🙃 这是内容 这是2级标题 这是内容 插入一个图片任意修改字体的颜色使得文章内容更加多彩神奇 插入一段代码 #include <iostream> using namespace std;int main() {cout << "Hello World!";return 0; }🛴

1-部署LVS-NAT

1.部署LVS-NAT 部署LVS-NAT集群主机名 IP地址client eth0:192.168.4.10/24proxy eth1:192.168.2.5/24eth0:192.168.4.5/24web1 eth0:192.168.2.100/24web2 eth0:192.168.2.200/24客户端访问LVS调度器的外网IP(VIP),调度器根据算法选择后端的一台真实服务器,将数据请求包转发…

3-LVS工作模式

3.LVS工作模式 NAT TUN DRhttps://blog.csdn.net/weixin_40470303/article/details/80541639 NAT 1.LVS服务器配两块网卡,一块连公网与用户通信,一块连内网与集群通信 2.负载路由器充当网关 3.支持端口映射,后端真实服务器的地址可能不是80,而是8080 4.集群节点处于一个…

Mybatis之动态sql

当你在业务中有需要通过传过来的条件来进行sql查询的时候,之前的手动拼接既麻烦又容易出错,动态sql就可以根据场景动态的构建查询。常用的动态sql标签if标签 <select id="selectAllBlog" parameterType= "map" resultType="Blog"> sele…