遥远的国度 (树链剖分换根),洛谷P3979
析:显然,若没有换根操作,则为树链剖分板子题,但是这道题我们考虑换根操作
考虑这样一个性质:在一棵树上,两点的距离路径是唯一的!!
也就是说,我们在修改路径上的点权时,不必考虑根在哪里,直接利用模板修改即可
麻烦的地方在于查询操作,我分了三种情况来讨论:(设 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的更多相关文章
- [模板] dfs序, 树链剖分, 换根
树链剖分 树链剖分是一种对树的分治, 可以把树上的任意一条链分解为 \(O(\log n)\) 条在dfs序上相邻的子链, 便于数据结构(如线段树)来维护. 另外, 子树在dfs序上也是一个连续的区间 ...
- 【树链剖分换根】P3979 遥远的国度
Description zcwwzdjn在追杀十分sb的zhx,而zhx逃入了一个遥远的国度.当zcwwzdjn准备进入遥远的国度继续追杀时,守护神RapiD阻拦了zcwwzdjn的去路,他需要zcw ...
- 洛谷P3979 遥远的国度 树链剖分+分类讨论
题意:给出一棵树,这棵树每个点有权值,然后有3种操作.操作一:修改树根为rt,操作二:修改u到v路径上点权值为w,操作三:询问以rt为根x子树的最小权值. 解法:如果没有修改树根操作那么这题就是树链剖 ...
- BZOJ 3083 遥远的国度(树链剖分+LCA)
Description 描述zcwwzdjn在追杀十分sb的zhx,而zhx逃入了一个遥远的国度.当zcwwzdjn准备进入遥远的国度继续追杀时,守护神RapiD阻拦了zcwwzdjn的去路,他需要z ...
- 【bzoj3083】遥远的国度 树链剖分+线段树
题目描述 描述zcwwzdjn在追杀十分sb的zhx,而zhx逃入了一个遥远的国度.当zcwwzdjn准备进入遥远的国度继续追杀时,守护神RapiD阻拦了zcwwzdjn的去路,他需要zcwwzdjn ...
- BZOJ 3083 遥远的国度 树链剖分
3083: 遥远的国度 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 797 Solved: 181[Submit][Status] Descrip ...
- BZOJ 3083 遥远的国度 树链剖分+线段树
有换根的树链剖分的裸题. 在换根的时候注意讨论. 注意数据范围要开unsigned int或longlong #include<iostream> #include<cstdio&g ...
- BZOJ 3083: 遥远的国度(树链剖分+DFS序)
可以很显而易见的看出,修改就是树链剖分,而询问就是在dfs出的线段树里查询最小值,但由于这道题会修改根节点,所以在查询的时候需判断x是否为root的祖先,如果不是就直接做,是的话应该查询从1-st[y ...
- BZOJ 3083: 遥远的国度 [树链剖分 DFS序 LCA]
3083: 遥远的国度 Time Limit: 10 Sec Memory Limit: 1280 MBSubmit: 3127 Solved: 795[Submit][Status][Discu ...
随机推荐
- 基于ABP落地领域驱动设计-06.正确区分领域逻辑和应用逻辑
目录 系列文章 领域逻辑和应用逻辑 多应用层 示例:正确区分应用逻辑和领域逻辑 学习帮助 系列文章 基于ABP落地领域驱动设计-00.目录和前言 基于ABP落地领域驱动设计-01.全景图 基于ABP落 ...
- 精尽Spring Boot源码分析 - 文章导读
该系列文章是笔者在学习 Spring Boot 过程中总结下来的,里面涉及到相关源码,可能对读者不太友好,请结合我的源码注释 Spring Boot 源码分析 GitHub 地址 进行阅读 Sprin ...
- .Net Core 常用开发工具(IDE和运行时、Visual Studio插件、Visual Studio Code插件)
IDE和运行时 组件名 描述 可选版本 推荐版本 Visual Studio Community 社区免费版 For Visual Studio 2017 For Visual Studio 2019 ...
- 流程自动化RPA,Power Automate Desktop系列 - DotNet Core打包并发布Nuget Package
一.背景 DotNet Core通常基于Nuget来实现包管理,如果你想要把自己的实现共享给其他人,通常我们需要把本地项目打包好,然后发布到对应的Nuget Server上,以便于其他人可以查找.安装 ...
- Redis的主从数据一致性
我们学习了 AOF 和 RDB,如果 Redis 发生了宕机,它们可以分别通过回放日志和重新读入 RDB 文件的方式恢复数据,从而保证尽量少丢失数据,提升可靠性.不过,即使用了这两种方法,也依然存在服 ...
- 万字长文肝Git--全流程知识点包您满意【建议收藏】
您好,我是码农飞哥,感谢您阅读本文,欢迎一键三连哦. 本文将首先介绍在本地搭建GitLab服务,然后重点介绍Git的常用命令,Git的核心概念以及冲突处理,最后介绍Git与SVN的区别 干货满满,建议 ...
- redis集群搭建中遇到的一些问题
redis单机模式启动后,修改完配置文件,使用以下命令创建redis集群: sudo ./src/redis-trib.rb create --replicas 1 ip1:6379 ip2:6379 ...
- mysql 去重的两种方式
1.distinct一般用于获取不重复字段的条数 使用原则: 1)distinct必须放在要查询字段的开头,不能放在查询字段的中间或者后面 select distinct name from user ...
- mysql 深度解析auto-increment自增列"Duliplicate key"问题
转载自:https://cloud.tencent.com/developer/article/1367681 问题描述 近期,线上有个重要Mysql客户的表在从5.6升级到5.7后master上插入 ...
- angular小记
declarations:包装组件或指令等 providers:依赖注入 imports:导入其他模块 bootstrap:设置根组件 exports:导出组件或指令等 app.component.t ...