BZOJ 3307 雨天的尾巴 (树上差分+线段树合并)
题目大意:给你一棵树,树上一共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 雨天的尾巴 (树上差分+线段树合并)的更多相关文章
- [BZOJ3307] 雨天的尾巴(树上差分+线段树合并)
[BZOJ3307] 雨天的尾巴(树上差分+线段树合并) 题面 给出一棵N个点的树,M次操作在链上加上某一种类别的物品,完成所有操作后,要求询问每个点上最多物品的类型. N, M≤100000 分析 ...
- [Luogu5327][ZJOI2019]语言(树上差分+线段树合并)
首先可以想到对每个点统计出所有经过它的链的并所包含的点数,然后可以直接得到答案.根据实现不同有下面几种方法.三个log:假如对每个点都存下经过它的链并S[x],那么每新加一条路径进来的时候,相当于在路 ...
- bzoj 3307: 雨天的尾巴【树剖lca+树上差分+线段树合并】
这居然是我第一次写线段树合并--所以我居然在合并的时候加点结果WAWAWAMLEMLEMLE--!ro的时候居然直接指到la就行-- 树上差分,每个点建一棵动态开点线段树,然后统计答案的时候合并即可 ...
- 2018.08.28 洛谷P4556 [Vani有约会]雨天的尾巴(树上差分+线段树合并)
传送门 要求维护每个点上出现次数最多的颜色. 对于每次修改,我们用树上差分的思想,然后线段树合并统计答案就行了. 注意颜色很大需要离散化. 代码: #include<bits/stdc++.h& ...
- bzoj3307 雨天的尾巴 题解(线段树合并+树上差分)
Description N个点,形成一个树状结构.有M次发放,每次选择两个点x,y 对于x到y的路径上(含x,y)每个点发一袋Z类型的物品.完成 所有发放后,每个点存放最多的是哪种物品. Input ...
- P4556 [Vani有约会]雨天的尾巴 /【模板】线段树合并 (树上差分+线段树合并)
显然的树上差分问题,最后要我们求每个点数量最多的物品,考虑对每个点建议线段树,查询子树时将线段树合并可以得到答案. 用动态开点的方式建立线段树,注意离散化. 1 #include<bits/st ...
- 【bzoj3307】雨天的尾巴 权值线段树合并
题目描述 N个点,形成一个树状结构.有M次发放,每次选择两个点x,y,对于x到y的路径上(含x,y)每个点发一袋Z类型的物品.完成所有发放后,每个点存放最多的是哪种物品. 输入 第一行数字N,M接下来 ...
- Luogu5327 ZJOI2019语言(树上差分+线段树合并)
暴力树剖做法显然,即使做到两个log也不那么优美. 考虑避免树剖做到一个log.那么容易想到树上差分,也即要对每个点统计所有经过他的路径产生的总贡献(显然就是所有这些路径端点所构成的斯坦纳树大小),并 ...
- [NOIP2016]天天爱跑步(树上差分+线段树合并)
将每个人跑步的路径拆分成x->lca,lca->y两条路径分别考虑: 对于在点i的观察点,这个人(s->t)能被观察到的充要条件为: 1.直向上的路径:w[i]=dep[s]-dep ...
随机推荐
- google浏览器中 查看记住的账号和密码
对于一个有“健忘症”的人来说,密码形同虚设..设置了就忘记,每次登陆都要重新设置密码... 然后,无意中发现,谷歌浏览器点过一次记住密码后,竟然可以明文查看账号和密码!! 步骤: 1.打开谷歌浏览器, ...
- Java基础学习总结(60)——Java常用的八种排序算法
1.直接插入排序 经常碰到这样一类排序问题:把新的数据插入到已经排好的数据列中. 将第一个数和第二个数排序,然后构成一个有序序列 将第三个数插入进去,构成一个新的有序序列. 对第四个数.第五个数--直 ...
- Oracle的JDBC Url的几种方式
1.普通SID方式jdbc:oracle:thin:username/password@x.x.x.1:1521:SID2.普通ServerName方式 jdbc:Oracle:thin:userna ...
- 对thinkpad太失望了
本来本着对thinkpad的信任买的,结果买回来一直吱吱吱吱响个不停. 好像是磁盘的问题,太垃圾了. http://benyouhui.it168.com/thread-1111376-1-1.htm ...
- Mybatis分页插件2.0版本号公布
项目地址:http://git.oschina.net/free/Mybatis_PageHelper 软件介绍:http://www.oschina.net/p/mybatis_pagehelper ...
- Eclipse开发C/C++程序的MinGw配置
环境变量设置: a.鼠标右击桌面"计算机"(WindowsXp是"我的电脑")->"属性" b.WindowsXP时,在新弹出的属性窗 ...
- 轻松学习JavaScript十二:JavaScript基于面向对象之创建对象(一)
这一次我们深入的学习一下JavaScript面向对象技术,在学习之前,必要的说明一下一些面向对象的一些术语. 这也是全部面对对象语言所拥有的共同点.有这样几个面向对象术语: 对象 ECMA-262把对 ...
- SOA概念具体解释
1.概述 1.1基本定义 SOA(Service-Oriented Architecture)既面向服务的体系结构,是一个组件模型.它将应用程序猿的不同功能可是(称为服务)通过定义良好的接口联系起来. ...
- iOS_第3方类库MBprogressHUD
1,将下载好的第3方类库MBprogressHUD源代码包增加到project(事实上就是一个.h和.m文件) 2,进入project的Build Phases,将源代码包里面的所有.m文件所有加入到 ...
- UVA 1541 - To Bet or Not To Bet 记忆化DP概率
Alexander Charles McMillan loves to gamble, and during his last trip to the casino he ran across a n ...