[题解] P4556 [Vani有约会]雨天的尾巴

·题目大意

给定一棵树,有m次修改操作,每次修改 \(( x\) \(y\) \(z )\) 表示 \((x,y)\) 之间的路径上数值 \(z\) 的个数 \(+1\) 。最后求每个节点上数量最多的数是哪个,如果有多个相同的则输出较小的。

·思路分析

想到用线段树合并+树上差分。由于值存储在点上,所以在 \(fa[LCA(x,y)],LCA(x,y)\) 上 \(-1\),在\(x,y\) 上 \(+1\) 即可。

注意线段树在 \(pushup()\) 时要记得更新是哪个数字数量最多。

·代码实现

#include <bits/stdc++.h>
#define reg register
using namespace std;
namespace io{
char ch[20];
template<typename T>inline void read(T &x){
x=0;
char ch,f=0;
while(!isdigit(ch=getchar()))f|=ch=='-';
while(isdigit(ch))x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
x=f?-x:x;
}
template<typename T>inline void write(T x){
(x<0)&&(x=-x,putchar('-'));
(x)||(putchar('0'));
reg int i=0;
while(x)ch[i++]=x%10^48,x/=10;
while(i)putchar(ch[--i]);
}
} //快读快写
#define rd io::read
#define wt io::write
using namespace std;
const int maxN=200100;
int son[maxN],dep[maxN],fa[maxN],top[maxN],rt[maxN],head[maxN];
int tot,d[maxN],n,m,ans[maxN],cnt;
stack<int>st;
struct Seg_Tree{
int ch[2],val,id;
}t[maxN<<5];
struct Edge{
int to,next;
}e[maxN<<1];
inline void add(int x,int y){e[++tot]={y,head[x]};head[x]=tot;}
void dfs1(int),dfs2(int,int),solve(int),pup(int),del(int);
int LCA(int,int),merge(int,int,int,int);
void update(int,int,int&,int,int);
int main(){
rd(n);rd(m);
for(reg int i=1;i<n;++i){
int x,y;
rd(x);rd(y);
add(x,y);add(y,x);
}
dfs1(1);memset(top,0,sizeof(top));
dfs2(1,1);
for(reg int i=1;i<=m;++i){
int x,y,z,k;
rd(x);rd(y);rd(z);
k=LCA(x,y);
update(1,100000,rt[fa[k]],z,-1);update(1,100000,rt[k],z,-1);
update(1,100000,rt[x],z,1);update(1,100000,rt[y],z,1);
}
solve(1);
for(reg int i=1;i<=n;++i){
wt(ans[i]);putchar('\n');
}
return 0;
}
void solve(int x){
for(reg int i=head[x];i;i=e[i].next){
int s=e[i].to;
if(s==fa[x])continue;
solve(s);
rt[x]=merge(1,100000,rt[x],rt[s]);
}
ans[x]=t[rt[x]].id;
if(t[rt[x]].val==0)ans[x]=0;
}
int merge(int l,int r,int x,int y){
if(!x||!y)return x+y;
if(l==r){
t[x].val+=t[y].val;
t[x].id=l;
return x;
}
int mid=l+r>>1;
t[x].ch[0]=merge(l,mid,t[x].ch[0],t[y].ch[0]);
t[x].ch[1]=merge(mid+1,r,t[x].ch[1],t[y].ch[1]);
pup(x);del(y);
return x;
}
void del(int x){t[x]={0,0,0};st.push(x);}
void pup(int x){
if(t[t[x].ch[0]].val>=t[t[x].ch[1]].val)t[x].val=t[t[x].ch[0]].val,t[x].id=t[t[x].ch[0]].id;
else t[x].val=t[t[x].ch[1]].val,t[x].id=t[t[x].ch[1]].id;
}
void update(int l,int r,int &k,int x,int v){
if(!k){
if(st.empty())k=++cnt;
else k=st.top(),st.pop();
}
if(l==r){
t[k].val+=v;t[k].id=l;
return;
}
int mid=l+r>>1;
if(x<=mid)update(l,mid,t[k].ch[0],x,v);
else update(mid+1,r,t[k].ch[1],x,v);
pup(k);
}
void dfs1(int x){
dep[x]=dep[fa[x]]+1;top[x]=1;
int mxson=-1;
for(reg int i=head[x];i;i=e[i].next){
int s=e[i].to;
if(dep[s])continue;
fa[s]=x;dfs1(s);top[x]+=top[s];
if(top[s]>mxson)mxson=top[s],son[x]=s;
}
}
void dfs2(int x,int u){
top[x]=u;
if(son[x])dfs2(son[x],u);
for(reg int i=head[x];i;i=e[i].next){
int s=e[i].to;
if(top[s])continue;
dfs2(s,s);
}
}
int LCA(int x,int y){
while(top[x]^top[y]){
if(dep[top[x]]<dep[top[y]])swap(x,y);
x=fa[top[x]];
}
return dep[x]<dep[y]?x:y;
}//树链剖分求LCA

[题解] P4556 [Vani有约会]雨天的尾巴的更多相关文章

  1. P4556 [Vani有约会]雨天的尾巴 (线段树合并)

    P4556 [Vani有约会]雨天的尾巴 题意: 首先村落里的一共有n座房屋,并形成一个树状结构.然后救济粮分m次发放,每次选择两个房屋(x,y),然后对于x到y的路径上(含x和y)每座房子里发放一袋 ...

  2. 洛谷 P4556 [Vani有约会]雨天的尾巴 解题报告

    P4556 [Vani有约会]雨天的尾巴 题目背景 深绘里一直很讨厌雨天. 灼热的天气穿透了前半个夏天,后来一场大雨和随之而来的洪水,浇灭了一切. 虽然深绘里家乡的小村落对洪水有着顽固的抵抗力,但也倒 ...

  3. P4556 [Vani有约会]雨天的尾巴(线段树合并+lca)

    P4556 [Vani有约会]雨天的尾巴 每个操作拆成4个进行树上差分,动态开点线段树维护每个点的操作. 离线处理完向上合并就好了 luogu倍增lca被卡了5分.....于是用rmq维护.... 常 ...

  4. 洛谷P4556 [Vani有约会]雨天的尾巴(线段树合并)

    题目背景 深绘里一直很讨厌雨天. 灼热的天气穿透了前半个夏天,后来一场大雨和随之而来的洪水,浇灭了一切. 虽然深绘里家乡的小村落对洪水有着顽固的抵抗力,但也倒了几座老房子,几棵老树被连根拔起,以及田地 ...

  5. P4556 [Vani有约会]雨天的尾巴

    目录 思路 优化 过程中的问题/疑问 错误 代码 思路 每个节点维护一课线段树(当然是动态开点) 线段树的作用是统计这个节点有多少种粮食型号,以及最多的粮食型号 然后树上差分,u和v点 +1,lca( ...

  6. 2018.08.28 洛谷P4556 [Vani有约会]雨天的尾巴(树上差分+线段树合并)

    传送门 要求维护每个点上出现次数最多的颜色. 对于每次修改,我们用树上差分的思想,然后线段树合并统计答案就行了. 注意颜色很大需要离散化. 代码: #include<bits/stdc++.h& ...

  7. 洛咕 P4556 [Vani有约会]雨天的尾巴

    终于把考试题清完了...又复活了... 树上差分,合并用线段树合并,但是空间会炸. 某大佬:lca和fa[lca]减得时候一定已经存在这个节点了,所以放进vector里,合并完之后减掉就好了... 玄 ...

  8. P4556 [Vani有约会]雨天的尾巴(线段树合并)

    传送门 一道线段树合并 首先不难看出树上差分 我们把每一次修改拆成四个,在\(u,v\)分别放上一个,在\(lca\)和\(fa[lca]\)各减去一个,那么只要统计一下子树里的总数即可 然而问题就在 ...

  9. P4556 [Vani有约会]雨天的尾巴 /【模板】线段树合并 (树上差分+线段树合并)

    显然的树上差分问题,最后要我们求每个点数量最多的物品,考虑对每个点建议线段树,查询子树时将线段树合并可以得到答案. 用动态开点的方式建立线段树,注意离散化. 1 #include<bits/st ...

随机推荐

  1. 拿来-util工具函数

    记录一些写的好的工具函数.以便学习和项目中直接拿来使用. 判断值是否相等:使用于任何数据类型:基本数据类型和复杂深层次对象 function deepEqual (a, b) { if (a === ...

  2. PHP的图片转base64,base64图片转换为图片并保存代码

    打卡记录 1. 图片转base64代码 /*图片转换为 base64格式编码*/ $img = 'images/avatar.jpg'; $base64_img = base64EncodeImage ...

  3. 【C#】C#中使用GDAL3(一):Windows下超详细编译C#版GDAL3.3.0(VS2015+.NET 4+32位/64位)

    转载请注明原文地址:https://www.cnblogs.com/litou/p/15004877.html 目录 一.介绍 二.编译准备 三.编译SQLite 四.编译LibTiff 五.编译PR ...

  4. GIS数据资源下载

    GeoJSON数据下载 1.全国.省.市.县级geojson数据下载 地址:http://datav.aliyun.com/tools/atlas/#&lat=33.5219039961561 ...

  5. 大数据学习(06)——Ozone介绍

    前面几篇文章把Hadoop常用的模块都学习了,剩下一个新模块Ozone,截止到今天最新版本是0.5.0Beta,还没出正式版.好在官方网站有文档,还是中文版的,但是中文版资料没有翻译完整,我试着把它都 ...

  6. 移植TensorFlow到Windows平台

    2015年11月,Google宣布开源旗下机器学习工具TensorFlow,引发业界热潮.TensorFlow原生支持*unix系和安卓平台,但并不提供对Windows平台的支持.如果想在Window ...

  7. 【Azure 应用服务】Azure Function HTTP 触发后, 230秒就超时。而其他方式触发的Function, 执行5分钟后也超时,如何调整超时时间?

    问题描述 Azure Function HTTP 触发后, 230秒就超时,而其他方式触发的Function, 执行5分钟后也超时,如何调整超时时间? 问题分析 查阅官方文档,对函数应用超时持续时间有 ...

  8. Podman 快速入门

    今天在某云上新购一台云服务器,发现已经有了 CentOS8.2 官方镜像可选,出于对新鲜事物的好奇,我决定开始采用 CentOS8.2,即使我还没有为它的新特性做好准备. 我的应用主要以单机版容器为主 ...

  9. 3.python编程与计算机的关系,如何执行python文件

    上一节预告了这一章想讲如何不停地和世界打招呼,这涉及到编程中一个重要的概念:循环. 但经过了两周断更后细想了一下,不行,我们得对上一章进行补充,而且这个补充非常关键!也印证了上一章所说的: 上一节章很 ...

  10. 用华为云cli,管理华为云服务器的,安全组端口

    ---[前言]--- 关键字 hcloud 华为 命令行 linux windows powershell 前些天,大家因为华为云,是否应该默认开启端口,大家吵起来了,所以我抽空写了此文.解决问题,缓 ...