题目大意:给你一棵树,树上一共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. CodeForces-546D Soldier and Number Game 筛法+动态规划

    题目链接:https://cn.vjudge.net/problem/CodeForces-546D 题意 抱歉,我给忘了,现在看题目又看不懂: P 思路 筛法+dp 话说这个函数应该是积性函数,然后 ...

  2. web前端知识框架

  3. 慎用PHP的unset、array_unique方法

    背景 在日常工作中,可能会经常遇到一些PHP的代码场景,需要我们去除数组中的某个项,通常会直接调用unset方法,但是如果用得不妥,会给自己挖坑 1.实操 以下使用具体例子进行证明假设有数组如下值: ...

  4. 马上着手开发ios应用程序

    https://developer.apple.com/library/ios/referencelibrary/GettingStarted/RoadMapiOSCh/chapters/Introd ...

  5. Ubuntu中的Docker搭建Tensorflow环境

    一.docker环境安装 1)更新.安装依赖包 sudo apt-get update sudo apt-get install apt-transport-https ca-certificates ...

  6. JavaScript(DOM编程补充一)

    HTML属性的直接调用: 还可以通过getAttribute方法获取属性的值: setAttribute方法: removeAttribute ---------------------------- ...

  7. eclipse svn -- - --- appears to be part of a subversion 1.7 or greater....解决方法

    安装与svn1.7相兼容的flex Eclipse中的SVN(subclipse) 今天差点被TortoiseSVN1.7和subclipse弄崩溃... 还好最后弄好了,在此把方法写出来,以免其他人 ...

  8. nodejs-路由(待补充)

    path Router 1 2 3 4 5 var express = require('express'); var Router = express.Router(); Router.get('/ ...

  9. BAT常问问题总结以及回答(问题汇总篇)

    几个大厂的面试题目目录: java基础(40题)https://www.cnblogs.com/television/p/9397968.html 多线程(51题) 设计模式(8点) JVM(12题) ...

  10. SQL从头開始

    SQL 分为两个部分:数据操作语言 (DML) 和 数据定义语言 (DDL) 查询和更新指令构成了 SQL 的 DML 部分: SELECT - 从数据库表中获取数据 UPDATE - 更新数据库表中 ...