洛谷 P2590 [ZJOI2008]树的统计
求赞
大家好,我非常喜欢暴力数据结构,于是我用块状树过了这道题目
题目:
一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w。
我们将以下面的形式来要求你对这棵树完成一些操作:
I. CHANGE u t : 把结点u的权值改为t
II. QMAX u v: 询问从点u到点v的路径上的节点的最大权值
III. QSUM u v: 询问从点u到点v的路径上的节点的权值和
注意:从点u到点v的路径上的节点包括u和v本身
我们可以将树大约划分为\(\sqrt{n}\)块,每个块内维护到块内根节点的路径长度以及点权最大值,而且,显然,我们可以通过寻找它们的\(LCA\)来找到他们路径上的有关信息,而这里我们已经对树进行了分块。
所以在同一个块内的暴跳时间复杂度最坏为\(O(\sqrt{n})\)
在块与块之间的暴跳的时间复杂度最坏为\(O(\sqrt{n})\)
轻松AC本题目
代码中有较详细注释,贴代码
#include<bits/stdc++.h>
using namespace std;
const int maxn=1e5+10;
struct cc{
int to,nex;
}e[maxn],dis[maxn];
int head[maxn],cnt1,h[maxn],cnt2;
void add1(int u,int v)//原树边
{
++cnt1;
e[cnt1].to=v;
e[cnt1].nex=head[u];
head[u]=cnt1;
}
void add2(int u,int v)//分块后块内树边
{
++cnt2;
dis[cnt2].to=v;
dis[cnt2].nex=h[u];
h[u]=cnt2;
}
int rt[maxn],mx[maxn],sum[maxn],siz[maxn];
int n,m,v[maxn],deep[maxn],len,fa[maxn];
void dfs(int u,int f,int dep)
{
deep[u]=dep;
int tmp=rt[u];
fa[u]=f;
for(int i=head[u];i;i=e[i].nex)
{
int v=e[i].to;
if(v!=f)
{
if(siz[tmp]+1<len)
{
add2(u,v);//块内树连边
rt[v]=tmp;
++siz[tmp];
}
dfs(v,u,dep+1);
}
}
}
void build(int u,int num,int vmx)//维护当前节点,到块内根节点的和,最大值
{
num+=v[u],sum[u]=num;
vmx=max(vmx,v[u]),mx[u]=vmx;
for(int i=h[u];i;i=dis[i].nex)
build(dis[i].to,num,vmx);
}
int query(int a,int b,int tag)
{
int ans1=0;//QSUM
int ans2=-(1<<30);//QMAX
while(a!=b)//类似于倍增,只不过这里的距离为sqrt(n)
{
if(deep[a]<deep[b]) swap(a,b);
if(rt[a]==rt[b])//若所属同一个块
{
ans1+=v[a];
ans2=max(ans2,v[a]);
a=fa[a];//由于在同一块内,暴力跳的复杂度只为O(sqrt(n))
}
else
{
if(deep[rt[a]]<deep[rt[b]]) swap(a,b);//块的深度更深
ans1+=sum[a];
ans2=max(ans2,mx[a]);
a=fa[rt[a]];//直接跳一个块
}
}
ans1+=v[a];
ans2=max(ans2,v[a]);//更新它们的LCA的值
if(tag==0) return ans2;
else return ans1;
}
void change(int u,int x)
{
v[u]=x;
if(u==rt[u]) build(u,0,-(1<<30));//如果是块内根节点就整个块更新
else build(u,sum[fa[u]],mx[fa[u]]);//如果不是,就从其父亲开始更新
}
int main()
{
int x,y;
scanf("%d",&n);
len=sqrt(n);
for(int i=1;i<n;++i)
scanf("%d%d",&x,&y),add1(x,y),add1(y,x);//原树边
for(int i=1;i<=n;++i)
scanf("%d",&v[i]),rt[i]=i;
dfs(1,0,0);
for(int i=1;i<=n;++i)
if(rt[i]==i)
build(i,0,-(1<<30));
scanf("%d",&m);
char opt[30];
for(int i=1;i<=m;++i)
{
scanf("%s%d%d",opt,&x,&y);
if(opt[1]=='M')//QMAX
printf("%d\n",query(x,y,0));//01维护询问问题
else if(opt[1]=='S')//QSUM
printf("%d\n",query(x,y,1));//01维护询问问题
else //CHANGE
change(x,y);
}
return 0;
}
骗分过样例,暴力出奇迹!!!
洛谷 P2590 [ZJOI2008]树的统计的更多相关文章
- 洛谷——P2590 [ZJOI2008]树的统计(树链剖分模板练手)
P2590 [ZJOI2008]树的统计 I. CHANGE u t : 把结点u的权值改为t II. QMAX u v: 询问从点u到点v的路径上的节点的最大权值 III. QSUM u v: 询问 ...
- 洛谷P2590 [ZJOI2008] 树的统计 [树链剖分]
题目传送门 树的统计 题目描述 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w. 我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t : 把结点u的权值改为t ...
- 洛谷P2590 [ZJOI2008]树的统计 题解 树链剖分+线段树
题目链接:https://www.luogu.org/problem/P2590 树链剖分模板题. 剖分过程要用到如下7个值: fa[u]:u的父节点编号: dep[u]:u的深度: size[u]: ...
- 洛谷 P2590 [ZJOI2008]树的统计(树链剖分)
题目描述一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w. 我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t : 把结点u的权值改为t II. QMAX u v ...
- [洛谷P2590][ZJOI2008]树的统计
题目大意:一棵树,支持三个操作, $CHANGE\;u\;t:$ 把结点$u$的权值改为$t$ $QMAX\;u\;v:$ 询问从点$u$到点$v$的路径上的节点的最大权值 $QSUM\;u\;v:$ ...
- 洛谷——P2590 [ZJOI2008]树的统计
https://www.luogu.org/problem/show?pid=2590#sub 题目描述 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w. 我们将以下面的形式来要求你对这 ...
- 洛谷.2590.[ZJOI2008]树的统计(树分块)
题目链接 Update:这种分块写法...可以被卡掉啊... 好像没有靠谱的树分块写法... /* 对树上节点进行分块,每个点记录dep,fa,val,Max,Sum,Max,Sum表示当前点在该块内 ...
- P2590 [ZJOI2008]树的统计(树链剖分)
P2590 [ZJOI2008]树的统计 虽然是入门树剖模板 但是我终于1A了(大哭) 懒得写啥了(逃 #include<iostream> #include<cstdio> ...
- P2590 [ZJOI2008]树的统计(LCT)
P2590 [ZJOI2008]树的统计 题目描述 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w. 我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t : 把 ...
随机推荐
- flex 输入框布局
1:创建一个弹性容器(display:flex) 2:构建2个或3个弹性项目. 3:把弹性项目设置为居中对齐.(align-items:center) 4:改变input自身对齐方式,把它设置为拉伸以 ...
- 原生JS替代jQuery的各种方法汇总
前端发展很快,现代浏览器原生 API 已经足够好用.我们并不需要为了操作 DOM.Event 等再学习一下 jQuery 的 API.同时由于 React.Angular.Vue 等框架的流行,直接操 ...
- Openlayer 3加载本地ArcGIS切片
第一篇博客,简单的开个头吧.希望自己能坚持记录.一般什么情况什么人需要这样的需求呢,伐木的光头强大哥说我们在深山老林里,没网的啊,地图就手机本地duang的加载一下吧.那么Server啊就要丢掉丢掉. ...
- 可达用户投资额的计算(Java)
有话要说: 前阵子遇到了一个计算可达用户投资额的问题,觉得非常有趣,故把它记录下来. 问题描述: 某产品可被投资,注册后才可以投资,其注册可以被邀请(并不是每个人都是被邀请的).邀请人可以邀请多个人注 ...
- Android 沉浸式状态栏完美解决方案
现在搜索Android 沉浸式状态栏,真的是一堆一堆,写的特别多,但是真正用的舒服的真没有,在这里自己整理一下开发记录 注意,在使用这个步骤过程之前,请把之前设置的代码注释一下 把布局带有androi ...
- Linux下载_Linux系统各种版本ISO镜像下载(redhat,centos,oracle,ubuntu,openSUSE)
以下是风哥收集的Linux系统各种版本ISO镜像下载,包括redhat,centos,oracle,ubuntu等linux操作系统. Linux下载1:红帽RedHat Linux(RHEL5.RH ...
- sql面试 查找每个班级的前5名学生(取分类数据的前几条数据)
关键字PARTITION BY 自己看代码喽~ SELECT * FROM ( SELECT ROW_NUMBER() OVER (PARTITION BY ClassType ORDER BY Sc ...
- python datetime操作
#datetime object转化为timestamp import datetime now = datetime.datetime.now() now_timestamp = time.mkti ...
- c/c++ 右值引用,forward关键字
c++ forward关键字 forward的由来:模板函数中的推导类型,作为另一函数的参数时,不管实参是什么类型,作为另一个参数的实参时,都变成了左值.因为C++里规定函数的形参就是左值,不过调用侧 ...
- 「技巧」如何快速安装 Sketch 插件
Sketch拥有强大丰富的插件,但是这些插件天各一方,四处查找下载地址非常麻烦.这里提供一个技巧,通过一个入口可以安装各种插件,基本涵盖了市面上所有靠谱的插件. 准备 Sketch54 Runner ...