首页 > 其他分享 >CF207C3 Game with Two Trees

CF207C3 Game with Two Trees

时间:2024-05-11 16:32:16浏览次数:27  
标签:ch 对应点 int CF207C3 Trees 我们 Game 重链 节点

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 号节点互相匹配,答案为 1
    for(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 998244353

const 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. 对应点要怎么求——》树剖加倍增

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

标签:ch,对应点,int,CF207C3,Trees,我们,Game,重链,节点
From: https://www.cnblogs.com/binbinbjl/p/18186715

相关文章

  • 题解:CF1956A Nene's Game
    这道题其实挺有意思,多测里面还套了个多测。思路就是用向量模拟删除过程,具体请看代码里的注释。#include<bits/stdc++.h>usingnamespacestd;intk,q,a[105];voidsolve(){ intn; cin>>n; vector<int>ve; for(inti=1;i<=n;i++)ve.push_back(i);//把每个人放到向量......
  • 一道DP(2024ICPC武汉邀请赛A)-shaking trees
    ShakingTrees题外话这题易哥哥跟我说这题的时候,点明了这题是关于高度\(100\)的\(O(n^3)\)或者\(O(n^4)\)的dp,还有提出切割点的概念使序列化。dp是真的,序列化也是真的。只能说易哥哥我的神。不过其实切割点是根据树形态而变的,之前一直以为是不变的,歪了好久。不知道是我没get到......
  • C. Game on Permutation
    链接:https://codeforces.com/problemset/problem/1860/C洛谷链接:https://www.luogu.com.cn/problem/CF1860C相关知识点复习:LIS最长上升子序列链接:https://blog.csdn.net/lxt_Lucia/article/details/81206439关键:这题的思路在于找到LIS长度为2的点,比如13254那么显然3,2是......
  • hgame2023 vm
    vm逆向hgame2023vm简单翻阅一下发现,sub_140001000里面是vm_init,sub_1400010B0是主要的vm部分查看vm_init部分,发现只知道cpu结构的大小以及初始值和24-32字节的结构,前24字节的结构未知,暂时还无法构建cpu的结构,需要更多的信息。分析下面两图的函数可以得知,opcode总共有8个,byt......
  • ACK One x OpenKruiseGame 全球游戏服多地域一致性交付最佳实践
    作者:刘秋阳、蔡靖前言在当今全球一体化的经济环境下,数字娱乐产业正日益成为文化和商业交流的有力代表。在此背景下大量游戏厂商尝试游戏出海并取得了令人瞩目的成绩,许多游戏以全球同服架构吸引着世界各地广泛的玩家群体。游戏全球化部署不仅扩大了单个产品的市场规模,而且提高了......
  • CF1956F Nene and the Passing Game 题解
    题目链接点击打开链接题目解法首先答案为连边之后连通块的个数把限制中的\(i,j\)分开,则\(i,j\)能传球当且仅当\(i+l_i\lej-l_j\)且\(j-r_j\lei+r_i\)这是一个二维偏序的形式先考虑怎么暴力做,考虑将\((i+l_i,0),\;(i-l_i,1)\)排序,然后按顺序加入如果碰到操作\(......
  • CF1951I Growing Trees
    MyBlogsCF1951IGrowingTrees首先考虑确定了\(x_i\)如何判定是否合法。可以很容易的找出这样一个必要条件:\(\foralli,x_i\leqk\)。这是两个点的情况,考虑点数更多的情况。手玩之后可以推广到:对于任意导出子图,要求其内部的边数\(\leqk(|S|-1)\)。这个条件也是充分的,证明......
  • JDK源码分析-TreeSet
    概述TreeSet是Java集合框架中用于存储唯一元素的树形数据结构,它实现了NavigableSet接口,这意味着TreeSet中的元素不仅是有序的,还支持一系列的导航方法。TreeSet的内部实现主要依赖于TreeMap,通过TreeMap的键来维护元素的排序。 类图从以上类图可以看到,TreeSet实现了三个接口,......
  • Games 101: 旋转矩阵
    旋转矩阵本文主要介绍了旋转矩阵的推导,分为两种方式:旋转坐标旋转坐标轴以下坐标系都是右手坐标系旋转坐标已知坐标点\(A(x_a,y_a)\),旋转\(\theta\)角后变为坐标点\(B(x_b,y_b)\),求解旋转矩阵.\[{\large\begin{align*}\begin{split}x_a&=r_a\cdotcos(\alpha)=......
  • AGC014D Black and White Trees
    传送门[AGC014D]BlackandWhiteTree给出一颗\(N\)个节点组成的树,每个节点都可以被染成白色或者黑色;有高桥(先手)和青木(后手)两个人————高桥可以把任意一个点染成白色,青木则可以把任意一个点染成黑色,每个点只可染色一次。重复上述操作直到所有点都被染色后,只执行一次执行......