析:显然,若没有换根操作,则为树链剖分板子题,但是这道题我们考虑换根操作

  考虑这样一个性质:在一棵树上,两点的距离路径是唯一的!!

  也就是说,我们在修改路径上的点权时,不必考虑根在哪里,直接利用模板修改即可

  麻烦的地方在于查询操作,我分了三种情况来讨论:(设 root 为当前的根,id 为当前要查询的编号)

  1.若 deep[id]>=deep[root] ,则跟对他没有影响,直接查询即可

  2.若 deep[id]<deep[root] ,刚开始我是求出两点的 LCA ,然后查询 (1,num[lca]) 和 (num[root]+size[root],n)

  但是,这样会出现一些问题,比如这张图:

  

  当 root=7,id==5 时,我们这样查询就会漏掉 6

  那么我们能不能这样写呢 ? 查询(1,num[root]-1) 和(num[root]+size[root],n);

  显然也是不能的(虽然这数据比较水,这样写会多得几分),如下图:

  

  当 root==4 ,id==2 时,这样查询就会多出来一个点 3

  这...........可如何是好?

  其实,应该有大佬已经发现了,无论是什么情况,我们查询的边界始终是 (1,num[son[lca]]-1)和 (num[son[lca]+size[son[lca]]] , n)

  所以,我们只需查询 root 和 id 的 lca 的重儿子即可(保证在同一条链上)

代码如下:

#include<bits/stdc++.h>
#define re register int
#define next nett
#define lc rt<<1
#define rc rt<<1|1
#define mid ((l+r)>>1)
#define ii inline int
#define iv inline void
using namespace std;
const int N=1e6+10;
struct CUN
{
int minn,lazy;
}use[N<<4];
int n,m,root,timi,tot,lcas;
int to[N<<1],next[N<<1],head[N],num[N],top[N],son[N],deep[N],fa[N],size[N],chu[N],zh[N];
inline int read()
{
int x=0,f=1;
char ch=getchar();
while(ch<'0'||ch>'9')
{
if(ch=='-')
f=0;
ch=getchar();
}
while(ch>='0'&&ch<='9')
{
x=(x<<1)+(x<<3)+(ch^48);
ch=getchar();
}
return (f)?x:(-x);
}
ii add(int x,int y)
{
to[++tot]=y;
next[tot]=head[x];
head[x]=tot;
}
iv dfs1(int st,int f)
{
size[st]=1;
fa[st]=f;
deep[st]=deep[f]+1;
for(re i=head[st];i;i=next[i])
{
int p=to[i];
if(p==f)
continue;
dfs1(p,st);
size[st]+=size[p];
son[st]=(size[son[st]]>size[p])?son[st]:p;
}
}
iv dfs2(int st,int t)
{
top[st]=t;
num[st]=++timi;
zh[timi]=chu[st];
if(!son[st])
return;
dfs2(son[st],t);
for(re i=head[st];i;i=next[i])
{
int p=to[i];
if(p==fa[st]||p==son[st])
continue;
dfs2(p,p);
}
}
iv pp(int rt)
{
use[rt].minn=min(use[lc].minn,use[rc].minn);
}
iv build(int rt,int l,int r)
{
if(l==r)
{
use[rt].minn=zh[l];
return;
}
build(lc,l,mid);
build(rc,mid+1,r);
pp(rt);
}
ii gett(int x,int y)
{
int fx=top[x],fy=top[y];
while(fx!=fy)
{
if(deep[fx]<deep[y])
swap(x,y),swap(fx,fy);
if(fa[fx]==y)
return fx;
x=fa[fx];
fx=top[x];
}
if(deep[x]>deep[y])
swap(x,y);
return son[x];
}
iv pd(int rt,int l,int r)
{
if(use[rt].lazy)
{
use[lc].minn=use[rt].lazy;
use[rc].minn=use[rt].lazy;
use[lc].lazy=use[rc].lazy=use[rt].lazy;
use[rt].lazy=0;
}
}
iv change(int rt,int l,int r,int L,int R,int z)
{
if(R<l||r<L||L>R) return ;
if(L<=l&&r<=R)
{
use[rt].minn=z;
use[rt].lazy=z;
return;
}
pd(rt,l,r);
if(mid>=L)
change(lc,l,mid,L,R,z);
if(mid<R)
change(rc,mid+1,r,L,R,z);
pp(rt);
}
ii query(int rt,int l,int r,int L,int R)
{
if(R<l||r<L||L>R) return 999999999;
if(L<=l&&r<=R)
return use[rt].minn;
pd(rt,l,r);
if(mid>=R)
return query(lc,l,mid,L,R);
if(mid<L)
return query(rc,mid+1,r,L,R);
return min(query(lc,l,mid,L,R),query(rc,mid+1,r,L,R));
}
iv upd(int x,int y,int z)
{
int fx=top[x],fy=top[y];
while(fx!=fy)
{
if(deep[fx]<deep[fy])
swap(x,y),swap(fx,fy);
change(1,1,n,num[fx],num[x],z);
x=fa[fx];
fx=top[x];
}
if(deep[x]>deep[y])
swap(x,y);
change(1,1,n,num[x],num[y],z);
}
int main()
{
n=read();
m=read();
int a,b,c,d,lca;
for(re i=1;i<n;i++)
{
a=read();
b=read();
add(a,b);
add(b,a);
}
for(re i=1;i<=n;i++)
chu[i]=read();
root=1;
dfs1(root,root);
dfs2(root,root);
build(1,1,n);
root=read();
while(m--)
{
a=read();
if(a==1)
{
root=read();
continue;
}
if(a==2)
{
b=read();
c=read();
d=read();
upd(b,c,d);
}
if(a==3)
{
b=read();
if(b==root)
{
printf("%d\n",query(1,1,n,1,n));
continue;
}
lcas=gett(b,root);
if(deep[b]>=deep[root])
printf("%d\n",query(1,1,n,num[b],num[b]+size[b]-1));
else if(fa[lcas]==b)
printf("%d\n",min(query(1,1,n,1,num[lcas]-1),query(1,1,n,num[lcas]+size[lcas],n)));
else
printf("%d\n",query(1,1,n,num[b],num[b]+size[b]-1));
continue;
} }
return 0;
}

  

遥远的国度 (树链剖分换根),洛谷P3979的更多相关文章

  1. [模板] dfs序, 树链剖分, 换根

    树链剖分 树链剖分是一种对树的分治, 可以把树上的任意一条链分解为 \(O(\log n)\) 条在dfs序上相邻的子链, 便于数据结构(如线段树)来维护. 另外, 子树在dfs序上也是一个连续的区间 ...

  2. 【树链剖分换根】P3979 遥远的国度

    Description zcwwzdjn在追杀十分sb的zhx,而zhx逃入了一个遥远的国度.当zcwwzdjn准备进入遥远的国度继续追杀时,守护神RapiD阻拦了zcwwzdjn的去路,他需要zcw ...

  3. 洛谷P3979 遥远的国度 树链剖分+分类讨论

    题意:给出一棵树,这棵树每个点有权值,然后有3种操作.操作一:修改树根为rt,操作二:修改u到v路径上点权值为w,操作三:询问以rt为根x子树的最小权值. 解法:如果没有修改树根操作那么这题就是树链剖 ...

  4. BZOJ 3083 遥远的国度(树链剖分+LCA)

    Description 描述zcwwzdjn在追杀十分sb的zhx,而zhx逃入了一个遥远的国度.当zcwwzdjn准备进入遥远的国度继续追杀时,守护神RapiD阻拦了zcwwzdjn的去路,他需要z ...

  5. 【bzoj3083】遥远的国度 树链剖分+线段树

    题目描述 描述zcwwzdjn在追杀十分sb的zhx,而zhx逃入了一个遥远的国度.当zcwwzdjn准备进入遥远的国度继续追杀时,守护神RapiD阻拦了zcwwzdjn的去路,他需要zcwwzdjn ...

  6. BZOJ 3083 遥远的国度 树链剖分

    3083: 遥远的国度 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 797  Solved: 181[Submit][Status] Descrip ...

  7. BZOJ 3083 遥远的国度 树链剖分+线段树

    有换根的树链剖分的裸题. 在换根的时候注意讨论. 注意数据范围要开unsigned int或longlong #include<iostream> #include<cstdio&g ...

  8. BZOJ 3083: 遥远的国度(树链剖分+DFS序)

    可以很显而易见的看出,修改就是树链剖分,而询问就是在dfs出的线段树里查询最小值,但由于这道题会修改根节点,所以在查询的时候需判断x是否为root的祖先,如果不是就直接做,是的话应该查询从1-st[y ...

  9. BZOJ 3083: 遥远的国度 [树链剖分 DFS序 LCA]

    3083: 遥远的国度 Time Limit: 10 Sec  Memory Limit: 1280 MBSubmit: 3127  Solved: 795[Submit][Status][Discu ...

随机推荐

  1. Project Reactor 响应式编程

    目录 一. 什么是响应式编程? 二. Project Reactor介绍 三. Reactor核心概念 Flux 1. just() 2. fromArray(),fromIterable()和 fr ...

  2. uniapp 打包IOS 更新AppStore版本

    Hello 你好,我是大粽子. 最近随着新版本UI的发布APP也随之更新,随之而来的也就是IOS程序提审步骤,这次我详细的截图了每一个步骤,如果你正好也需要那么跟着我的节奏一步步来肯定是没问题的. 提 ...

  3. 1、linux集群服务器规划

  4. centos 8.3关闭防火墙,设置开机启动禁用防火墙

    关闭防火墙 [root@bogon ~]# systemctl stop firewalld      开机禁用防火墙 [root@bogon ~]# systemctl disable firewa ...

  5. K8S(Kubernetes)学习笔记

    Kubernetes(k8s)是google提供的开源的容器集群管理系统,在Docker技术的基础上,为容器化的应用提供部署运行.资源调度.服务发现和动态伸缩等一系列完整功能,提高了大规模容器集群管理 ...

  6. leetcode 1122

    思路分析: 主要思想:计数排序 先遍历arr1,然后计数,再遍历arr2时同时又排完序了,再继续把arr2不存在的数字,再遍历加到数组后面,也同时排完序了.方便快捷

  7. challenge

    一.查壳 无壳的64位linux文件 二.拖入ida静态分析 F5反编译一下. 发现这段代码实际意义就是把字符串传进v7当中.v7的值就是flag值. 这个函数整体分析下来,各种递归,尝试用c语言来复 ...

  8. LVM磁盘管理实战案例

    前言:这是关于centos/linux上的LVM磁盘上的扩容,缩减,磁盘类型等收集做的案例 至于ubuntu系统的磁盘扩容参考:https://www.jianshu.com/p/5dcfcec687 ...

  9. NTFS安全权限

    一.NTFS权限概述 1.通过设置NTFS权限,实现不同的用户访问不同的对象的权限 2.分配了真确的访问权限后,用户才能访问其资源 3.设置权限防止资源被篡改.删除 二.文件系统概述 文件系统即在外部 ...

  10. 嵌入式Redis服务器在Spring Boot测试中的使用

    1.概述 Spring Data Redis提供了一种与Redis实例集成的简单方法. 但是,在某些情况下,使用嵌入式服务器比使用真实服务器创建开发和测试环境更方便. 因此,我们将学习如何设置和使用嵌 ...