题目大意:给你一棵树,树上一共n个节点,共m次操作,每次操作给一条链上的所有节点分配一个权值,求所有节点被分配到所有的权值里,出现次数最多的权值是多少,如果出现次数相同就输出最小的。

(我辣鸡bzoj的权限号,洛谷上P4556也有这道题)

线段树合并入门题

也是比较常规的树上链的点差分 每次操作都在x,y上+1,在lca(x,y),fa[lca(x,y)]上-1

然后对每个点的所有差分操作构建一颗动态开点线段树,然后从叶节点向上合并即可

特别的,只有线段树的最底层存的是实际打的差分,而上层节点仅仅是用来 像分治一样在log(n)时间内 快速求得答案,所以最底层和上层所维护的东西也不一样。所以合并的过程中,两棵树的最底层节点是直接相加,而上层是通过下层来更新答案,而并非对这两个线段树直接合并,需要仔细思考。

总结:树上线段树合并 恰好和 树上主席树 相反

树上线段树合并是回溯时由子节点更新父节点的线段树

而树上主席树是通过深搜从父节点更新子节点的线段树

 #include <cstdio>
#include <algorithm>
#include <cstring>
#include <vector>
#define inf 0x3f3f3f3f
#define ll long long
#define N 100100
#define maxn 1000000
using namespace std;
//re
int n,m,cte,tot,num;
int head[N],dep[N],fa[N],tp[N],sz[N],son[N],root[N],a[N],ans[N];
int ls[N*],rs[N*];
struct Ques{
int x,y,w,ff;
}ques[N];
struct node{
int id,sum;
}ma[N*];
struct EDGE{
int to,nxt;
}edge[N*];
void ae(int u,int v)
{
cte++;
edge[cte].to=v;
edge[cte].nxt=head[u];
head[u]=cte;
}
int gc()
{
int rett=,fh=;char p=getchar();
while(p<''||p>'') {if(fh=='-')fh=;p=getchar();}
while(p>=''&&p<='') {rett=(rett<<)+(rett<<)+p-'';p=getchar();}
return rett*fh;
}
void tcs_dfs1(int x,int dad)
{
for(int j=head[x];j!=-;j=edge[j].nxt){
int v=edge[j].to;
if(v==dad) continue;
dep[v]=dep[x]+,fa[v]=x;
tcs_dfs1(v,x),sz[x]+=sz[v];
son[x]=(sz[v]>sz[son[x]])?v:son[x];
}sz[x]++;
}
void tcs_dfs2(int x)
{
root[x]=++tot;
if(son[x]) tp[son[x]]=tp[x],tcs_dfs2(son[x]);
for(int j=head[x];j!=-;j=edge[j].nxt){
int v=edge[j].to;
if(v==fa[x]||v==son[x]) continue;
tp[v]=v,tcs_dfs2(v);
}
}
int LCA(int x,int y)
{
while(tp[x]!=tp[y]){
if(dep[tp[x]]<dep[tp[y]]) swap(x,y);
x=fa[tp[x]];
}return dep[x]<dep[y]?x:y;
}
void seg_modify(int x,int l,int r,int rt,int w)
{
if(l==r){ma[rt].sum+=w,ma[rt].id=l;return;}
int mid=(l+r)>>;
if(x<=mid) seg_modify(x,l,mid,ls[rt]?ls[rt]:(ls[rt]=++tot),w);
else seg_modify(x,mid+,r,rs[rt]?rs[rt]:(rs[rt]=++tot),w);
if(ma[ls[rt]].sum>ma[rs[rt]].sum) ma[rt].id=ma[ls[rt]].id;
else if(ma[ls[rt]].sum<ma[rs[rt]].sum) ma[rt].id=ma[rs[rt]].id;
else ma[rt].id=min(ma[ls[rt]].id,ma[rs[rt]].id);
ma[rt].sum=max(ma[ls[rt]].sum,ma[rs[rt]].sum);
}
int seg_merge(int rx,int ry,int l,int r)
{
if(!rx||!ry) return rx+ry;
if(l==r){
ma[rx].sum+=ma[ry].sum;ma[rx].id=l;
return rx;}
int mid=(l+r)>>;
ls[rx]=seg_merge(ls[rx],ls[ry],l,mid);
rs[rx]=seg_merge(rs[rx],rs[ry],mid+,r);
if(ma[ls[rx]].sum>ma[rs[rx]].sum) ma[rx].id=ma[ls[rx]].id;
else if(ma[ls[rx]].sum<ma[rs[rx]].sum) ma[rx].id=ma[rs[rx]].id;
else ma[rx].id=min(ma[ls[rx]].id,ma[rs[rx]].id);
ma[rx].sum=max(ma[ls[rx]].sum,ma[rs[rx]].sum);
return rx;
}
void dfs_ans(int x)
{
for(int j=head[x];j!=-;j=edge[j].nxt){
int v=edge[j].to;
if(v==fa[x]) continue;
dfs_ans(v);
seg_merge(root[x],root[v],,num);
}
ans[x]=ma[root[x]].id;
} int main()
{
//freopen("data.in","r",stdin);
scanf("%d%d",&n,&m);
memset(head,-,sizeof(head));
int x,y,z,ff;
for(int i=;i<n;i++)
x=gc(),y=gc(),ae(x,y),ae(y,x);
dep[]=,tcs_dfs1(,-);
tp[]=,tcs_dfs2();
for(int i=;i<=m;i++)
{
ques[i].x=gc(),ques[i].y=gc();
ques[i].w=gc(),ques[i].ff=LCA(ques[i].x,ques[i].y);
a[++num]=ques[i].w;
}
sort(a+,a+num+);
num=unique(a+,a+num+)-(a+);
for(int i=;i<=m;i++)
{
int ww=lower_bound(a+,a+num+,ques[i].w)-a;
seg_modify(ww,,num,root[ques[i].x],);
seg_modify(ww,,num,root[ques[i].y],);
seg_modify(ww,,num,root[ques[i].ff],-);
if(fa[ques[i].ff]) seg_modify(ww,,num,root[fa[ques[i].ff]],-);
}
dfs_ans();
for(int i=;i<=n;i++) printf("%d\n",a[ans[i]]);
return ;
}

BZOJ 3307 雨天的尾巴 (树上差分+线段树合并)的更多相关文章

  1. [BZOJ3307] 雨天的尾巴(树上差分+线段树合并)

    [BZOJ3307] 雨天的尾巴(树上差分+线段树合并) 题面 给出一棵N个点的树,M次操作在链上加上某一种类别的物品,完成所有操作后,要求询问每个点上最多物品的类型. N, M≤100000 分析 ...

  2. [Luogu5327][ZJOI2019]语言(树上差分+线段树合并)

    首先可以想到对每个点统计出所有经过它的链的并所包含的点数,然后可以直接得到答案.根据实现不同有下面几种方法.三个log:假如对每个点都存下经过它的链并S[x],那么每新加一条路径进来的时候,相当于在路 ...

  3. bzoj 3307: 雨天的尾巴【树剖lca+树上差分+线段树合并】

    这居然是我第一次写线段树合并--所以我居然在合并的时候加点结果WAWAWAMLEMLEMLE--!ro的时候居然直接指到la就行-- 树上差分,每个点建一棵动态开点线段树,然后统计答案的时候合并即可 ...

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

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

  5. bzoj3307 雨天的尾巴 题解(线段树合并+树上差分)

    Description N个点,形成一个树状结构.有M次发放,每次选择两个点x,y 对于x到y的路径上(含x,y)每个点发一袋Z类型的物品.完成 所有发放后,每个点存放最多的是哪种物品. Input ...

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

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

  7. 【bzoj3307】雨天的尾巴 权值线段树合并

    题目描述 N个点,形成一个树状结构.有M次发放,每次选择两个点x,y,对于x到y的路径上(含x,y)每个点发一袋Z类型的物品.完成所有发放后,每个点存放最多的是哪种物品. 输入 第一行数字N,M接下来 ...

  8. Luogu5327 ZJOI2019语言(树上差分+线段树合并)

    暴力树剖做法显然,即使做到两个log也不那么优美. 考虑避免树剖做到一个log.那么容易想到树上差分,也即要对每个点统计所有经过他的路径产生的总贡献(显然就是所有这些路径端点所构成的斯坦纳树大小),并 ...

  9. [NOIP2016]天天爱跑步(树上差分+线段树合并)

    将每个人跑步的路径拆分成x->lca,lca->y两条路径分别考虑: 对于在点i的观察点,这个人(s->t)能被观察到的充要条件为: 1.直向上的路径:w[i]=dep[s]-dep ...

随机推荐

  1. Vue学习之路第十五篇:v-if和v-show指令

    1.v-if和v-show都是用来实现条件判断的指令. 2.先看代码 <body> <div id="app"> <button @click=&qu ...

  2. Echarts堆积柱状图排序问题

    Echarts堆积柱状图排序是按照堆积柱状图的柱子高度进行从大到小(或者从小到大)进行排序,方便查阅各坐标情况.以下是我自己研发的方法,有不对的地方敬请谅解,随时欢迎指教. 排序后效果如下图: (1) ...

  3. [BZOJ4916]神犇(Monster_Qi)和蒟蒻(SWHsz)

    很久很久以前,有一只神犇叫Monster_Qi; 很久很久之后,有一只蒟蒻叫SWHsz; 1<=N<=1E9,A.B模1E9+7; 求这个. 求μ的话直接输出1就行了因为除了1的平方外都有 ...

  4. Python docs

    刚刚发现了Python好用的文档查询网页:Python docs,可以选择python版本查询,跟OpenCV docs很相似,很好用-

  5. Strtus配置Tomcat出现问题2

    启动myeclipse的tomcat6.0有如下提示:The APR based Apache Tomcat Native library which allows optimal performan ...

  6. B - Networking

    B - Networking 思路:并查集板子. #include<cstdio> #include<cstring> #include<iostream> #in ...

  7. xcode对照两个分支中同一个文件

    对于同一个项目的两个分支,由于两个分支可能各自都做了一些改动.所以通过Source Control中的History...功能是无法查看的.例如以下图: 这个时候.我们须要用到xcode的另外一个功能 ...

  8. Mybatis分页插件2.0版本号公布

    项目地址:http://git.oschina.net/free/Mybatis_PageHelper 软件介绍:http://www.oschina.net/p/mybatis_pagehelper ...

  9. [C++设计模式] strategy 策略模式

    依照陈硕老师的观点.c++里有面向过程编程.面向对象编程,基于对象编程(面向接口编程)和泛型编程.四种思路都各有其适用场景. 面向过程编程是沿袭C的结构化编程思路,OOP是C++的核心,也是现代高级编 ...

  10. php面向对象之get和set方法

    php面向对象之get和set方法 简介 1.自己写get或者set 2.用系统的魔术方法__get和__set 代码 <?php class Person{ private $userName ...