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

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

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

  麻烦的地方在于查询操作,我分了三种情况来讨论:(设 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. 学习JDK源码(一):String

    用了好久的Java了,从来没有看过jdk的源码,趁着今天有点时间,拿出了jdk的源码看了下,今天先看了关于String的,毕竟开发中String类型使用最广泛.在我们下载安装jdk的时候,部分源码也已 ...

  2. 用vue ui创建的项目怎么关闭eslint校验

    在Vue Cli的控制面板找到配置-ESLint configuration,然后关闭保存时检查就可以了

  3. mysql的主从复制延迟问题--看这一篇就够了

    ​ 在之前我们已经讲解了一主一从,双主双从的mysql集群搭建,在单机应用的时候看起来没有问题,但是在企业的生产环境中,在很多情况下都会有复制延迟的问题. ​ 主从复制的原理我们在此处就不再赘述了,之 ...

  4. Mongo集群搭建

    1.集群角色及架构 在搭建集群之前,需要首先了解几个概念:路由,分片.副本集.配置服务器等 mongos,数据库集群请求的入口,所有的请求都通过mongos进行协调,不需要在应用程序添加一个路由选择器 ...

  5. SpringCloud Alibaba实战(11:引入服务网关Gateway)

    源码地址:https://gitee.com/fighter3/eshop-project.git 持续更新中-- 大家好,我是三分恶. 在前面的章节中,我们已经完成了服务间的调用.统一配置等等,在这 ...

  6. 学会这些CSS技巧让你写样式更加丝滑

    目录 1,前言 1,calc() 2,min() 3,max() 4,clamp() 5,gap 6,writing-mode 1,前言 记录一些很好用的css属性 1,calc() calc()函数 ...

  7. 增强采样软件PLUMED的安装与使用

    技术背景 增强采样(Enhanced Sampling)是一种在分子动力学模拟中常用的技术,其作用是帮助我们更加快速的在时间轴上找到尽可能多的体系结构及其对应的能量.比如一个氢气的燃烧反应,在中间过程 ...

  8. Mybatis学习(1)开发环境搭建

    什么是mybatis MyBatis是支持普通SQL查询,存储过程和高级映射的优秀持久层框架.MyBatis消除了几乎所有的JDBC代码和参数的手工设置以及结果集的检索.MyBatis使用简单的XML ...

  9. springCloud--admin监控使用

    Admin监控应用 Spring Boot提供的监控接口,例如:/health./info等等,实际上除了之前提到的信息,还有其他信息业需要监控:当前处于活跃状态的会话数量.当前应用的并发数.延迟以及 ...

  10. Thread对象 既传入了Runnable对象又重写了run()方法

    Thread本身是一个类,其run(): 正常情况下, target(即传入的new Runnable()对象)不为空,就是执行target里面的run方法 但是,重新run()方法之后,就不会在执行 ...