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

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

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

  麻烦的地方在于查询操作,我分了三种情况来讨论:(设 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. 基于ABP落地领域驱动设计-06.正确区分领域逻辑和应用逻辑

    目录 系列文章 领域逻辑和应用逻辑 多应用层 示例:正确区分应用逻辑和领域逻辑 学习帮助 系列文章 基于ABP落地领域驱动设计-00.目录和前言 基于ABP落地领域驱动设计-01.全景图 基于ABP落 ...

  2. 精尽Spring Boot源码分析 - 文章导读

    该系列文章是笔者在学习 Spring Boot 过程中总结下来的,里面涉及到相关源码,可能对读者不太友好,请结合我的源码注释 Spring Boot 源码分析 GitHub 地址 进行阅读 Sprin ...

  3. .Net Core 常用开发工具(IDE和运行时、Visual Studio插件、Visual Studio Code插件)

    IDE和运行时 组件名 描述 可选版本 推荐版本 Visual Studio Community 社区免费版 For Visual Studio 2017 For Visual Studio 2019 ...

  4. 流程自动化RPA,Power Automate Desktop系列 - DotNet Core打包并发布Nuget Package

    一.背景 DotNet Core通常基于Nuget来实现包管理,如果你想要把自己的实现共享给其他人,通常我们需要把本地项目打包好,然后发布到对应的Nuget Server上,以便于其他人可以查找.安装 ...

  5. Redis的主从数据一致性

    我们学习了 AOF 和 RDB,如果 Redis 发生了宕机,它们可以分别通过回放日志和重新读入 RDB 文件的方式恢复数据,从而保证尽量少丢失数据,提升可靠性.不过,即使用了这两种方法,也依然存在服 ...

  6. 万字长文肝Git--全流程知识点包您满意【建议收藏】

    您好,我是码农飞哥,感谢您阅读本文,欢迎一键三连哦. 本文将首先介绍在本地搭建GitLab服务,然后重点介绍Git的常用命令,Git的核心概念以及冲突处理,最后介绍Git与SVN的区别 干货满满,建议 ...

  7. redis集群搭建中遇到的一些问题

    redis单机模式启动后,修改完配置文件,使用以下命令创建redis集群: sudo ./src/redis-trib.rb create --replicas 1 ip1:6379 ip2:6379 ...

  8. mysql 去重的两种方式

    1.distinct一般用于获取不重复字段的条数 使用原则: 1)distinct必须放在要查询字段的开头,不能放在查询字段的中间或者后面 select distinct name from user ...

  9. mysql 深度解析auto-increment自增列"Duliplicate key"问题

    转载自:https://cloud.tencent.com/developer/article/1367681 问题描述 近期,线上有个重要Mysql客户的表在从5.6升级到5.7后master上插入 ...

  10. angular小记

    declarations:包装组件或指令等 providers:依赖注入 imports:导入其他模块 bootstrap:设置根组件 exports:导出组件或指令等 app.component.t ...