题目大意:给你一棵树,树上一共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. 【JavaScript框架封装】实现一个类似于JQuery的DOM框架的封装

    // DOM框架(选择器框架) (function (xframe) { // 需要参与链式访问的(必须使用prototype的方式来给对象扩充方法) xframe.extend({ /** * 向现 ...

  2. [LUOGU2730] 魔板

    搜索水题.因为只有8个数,排列一共有40320种,直接bfs,判重就行了. 但是判重的时候直接用8进制表示的话要88的bool数组.这种操作太low了,于是我们可以用康托展开,降成8!. 康托展开其实 ...

  3. 微信小程序 Image 图片实现宽度100%,高度自适应

    做法如下: 样式设置宽度100%, .img{ width:100%;} 添加属性 mode="widthFix", <image class="img" ...

  4. CCPC2018秦皇岛游记

    Day1 27号晚上8点多的火车. 然后..第二天(28号)6点40左右的样子到了天津(中转站) 然后一顿乱拍. 看到宝葫芦了没:) 然后.看到了狗不理包子铺...不过当时没开门,就溜了. 然后去秦皇 ...

  5. MySQL多线程数据导入导出工具Mydumper

    http://afei2.sinaapp.com/?p=456 今天在线上使用mysqldump将数据表从一个库导入到另外一个库,结果速度特别慢,印象中有个多线程的数据导入导出工具Mydumper,于 ...

  6. 使用SeaJS,require加载Jquery的时候总是为null

    这个问题困扰了我两天,使用别人的例子.官网down下来的example都没有问题.但是放到自己项目里就 var $=require("jquery") 为null. 后来发现,jq ...

  7. 洛谷——P1802 5倍经验日

    https://www.luogu.org/problem/show?pid=1802#sub 题目背景 现在乐斗有活动了!每打一个人可以获得5倍经验!absi2011却无奈的看着那一些比他等级高的好 ...

  8. POJ 2110

    终于过了,SHIT,二分+DFS即可,二分区间,根据数字是否在区间内,变成迷宫题了.枚举第一个格子,二分上界,即可. #include <iostream> #include <cs ...

  9. SqlCommand.DeriveParameters failed

    错误信息例如以下: SqlCommand.DeriveParameters failed because the SqlCommand.CommandText property value is an ...

  10. Hit 2255 Not Fibonacci

    今天下午刚起来眼睛就比較涨,,并且还有点恶心,唉.结果一直不在状态.并且这个题太坑了.. .. 点击此处即可传送 Hit 2255 Maybe ACMers of HIT are always fon ...