C. 雨天的尾巴

题目描述

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

输入格式

第一行数字N,M
接下来N-1行,每行两个数字a,b,表示a与b间有一条边
再接下来M行,每行三个数字x,y,z.如题

输出格式

输出有N行
每i行的数字表示第i个点存放最多的物品是哪一种,如果有
多种物品的数量一样,输出编号最小的。如果某个点没有物品则输出0

样例

样例输入


样例输出


数据范围与提示

1<=N,M<=100000
1<=a,b,x,y<=N
1<=z<=10910^910​9​​

暴力能得50分呢……

树上操作首先会想到树剖和树上差分吧,这里只说差分;

离线处理,权值线段树维护每一个点的状态(每种物品出现次数及其最大值),对于每次操作,将x+1,y+1,LCA(x,y)-1,fa[LCA]-1最后dfs合并线段树统计答案即可。

注意合并(修改)叶子节点时最大值是加而不是取max。

这道题比较恶心的是卡内存,卡了我四节课…

如果线段树合并操作是建新节点的话会MLE,代码如下:

int merge(int x,int y)
{
if(!x||!y)return x+y;
int now=++cnt;
sum(now)=sum(x)+sum(y);
l(now)=merge(l(x),l(y));
r(now)=merge(r(x),r(y));
if(!l(x) && !r(x))maxn(now)=maxn(x)+maxn(y);
else maxn(now)=max( maxn(l(now)) , maxn(r(now)) );
return now;
}

但是显然不这样的话数据会出错(将y的子树同时变为x的子树,之后在合并x时会修改数据),但是其实并不需要让线段树最后是正确的,只需要在y数据发生错误之前记录答案即可。

标程

#include<iostream>
#include<cstdio>
#include<map>
#include<time.h>
#include<cstdlib>
#include<algorithm>
using namespace std;
struct edge
{
int u,v,next;
#define u(x) ed[x].u
#define v(x) ed[x].v
#define n(x) ed[x].next
}ed[];
int first[],num_e;
#define f(x) first[x]
int n,m,Q,fa[][],bin[],dep[];
int x[],y[],z[],z2[];
map<int,int> mp;
int mmp[];
int ans[]; struct tree
{
int l,r,sum,maxn;
#define l(x) tr[x].l
#define r(x) tr[x].r
#define sum(x) tr[x].sum
#define maxn(x) tr[x].maxn
}tr[];
int cnt,rt[]; int ask(int l,int r,int a)
{
if(sum(a)==)return ;
if(l==r)return l;
int mid=(l+r)>>;
if(maxn(l(a))>=maxn(r(a)))return ask(l,mid,l(a));
return ask(mid+,r,r(a));
}
void add(int &mark,int l,int r,int loc,int val)
{
if(!mark)mark=++cnt;
if(l==r){sum(mark)+=val;maxn(mark)+=val;return;}
int mid=(l+r)>>;
if(loc<=mid)add(l(mark),l,mid,loc,val);
else add(r(mark),mid+,r,loc,val);
sum(mark)=sum(l(mark))+sum(r(mark));
maxn(mark)=max( maxn(l(mark)) , maxn(r(mark)));
}
int merge(int x,int y)
{
if(!x||!y)return x+y;
l(x)=merge(l(x),l(y));
r(x)=merge(r(x),r(y));
sum(x)=sum(x)+sum(y);
if(!l(x) && !r(x))maxn(x)=maxn(x)+maxn(y);
else maxn(x)=max( maxn(l(x)) , maxn(r(x)) );
return x;
}
void dfs2(int x,int ffa);
inline int read();
int LCA(int x,int y);
void dfs(int x,int ffa);
inline void add_e(int u,int v);
signed main()
{
// freopen("4.in","r",stdin);
// freopen("out.txt","w",stdout); bin[]=;
for(int i=;i<=;i++)bin[i]=bin[i-]*;
n=read(),Q=read();
int ta,tb;
for(int i=;i<n;i++)
{
ta=read(),tb=read();
add_e(ta,tb);
add_e(tb,ta);
}
for(int j=;j<=Q;j++)
x[j]=read(),y[j]=read(),z[j]=read(),z2[j]=z[j];
sort(z2+,z2+Q+);
m=unique(z2+,z2+Q+)-z2-;
for(int i=;i<=Q;i++)
{
int loc=lower_bound(z2+,z2+m+,z[i])-z2;
mp[z[i]]=loc;
mmp[loc]=z[i];
}
dfs(,);
for(int j=;j<;j++)
for(int i=;i<=n;i++)
fa[i][j]=fa[fa[i][j-]][j-];
mmp[]=;
for(int i=;i<=Q;i++)
{
int loc=mp[z[i]],
lca=LCA(x[i],y[i]),
ffa=fa[lca][];
add(rt[x[i]],,m,loc,);
add(rt[y[i]],,m,loc,);
add(rt[lca], ,m,loc,-);
if(ffa)
add(rt[ffa] ,,m,loc,-);
}
dfs2(,);
ans[]=ask(,m,rt[]);
for(int i=;i<=n;i++)
printf("%d\n",mmp[ans[i]]);
}
void dfs2(int x,int ffa)
{
for(int i=f(x);i;i=n(i))
if(v(i)!=ffa)
{
dfs2(v(i),x);
ans[v(i)]=ask(,m,rt[v(i)]);
rt[x]=merge(rt[x],rt[v(i)]);
}
}
inline int read()
{
int s=;char a=getchar();
while(a<''||a>'')a=getchar();
while(a>=''&&a<=''){s=s*+a-'';a=getchar();}
return s;
}
inline void add_e(int u,int v)
{
++num_e;
u(num_e)=u;
v(num_e)=v;
n(num_e)=f(u);
f(u)=num_e;
}
void dfs(int x,int ffa)
{
fa[x][]=ffa;
dep[x]=dep[ffa]+;
for(int i=f(x);i;i=n(i))
if(v(i)!=ffa)
dfs(v(i),x);
}
int LCA(int x,int y)
{
if(dep[x]>dep[y])swap(x,y);
while(dep[x]!=dep[y])
for(int i=;;i++)
if(dep[fa[y][i]]<dep[x])
{
y=fa[y][i-];
break;
}
if(x==y)return x;
while(fa[x][]!=fa[y][])
for(int i=;;i++)
if(fa[x][i]==fa[y][i])
{x=fa[x][i-],y=fa[y][i-];break;}
return fa[x][];
}

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

  1. bzoj 3307: 雨天的尾巴 线段树合并

    题目大意: N个点,形成一个树状结构.有M次发放,每次选择两个点x,y对于x到y的路径上(含x,y)每个点发一袋Z类型的物品.问完成所有发放后,每个点存放最多的是哪种物品. 题解: 首先我们为每一个节 ...

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

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

  3. BZOJ_3307_雨天的尾巴_线段树合并+树上差分

    BZOJ_3307_雨天的尾巴_线段树合并 Description N个点,形成一个树状结构.有M次发放,每次选择两个点x,y 对于x到y的路径上(含x,y)每个点发一袋Z类型的物品.完成 所有发放后 ...

  4. [Vani有约会]雨天的尾巴 线段树合并

    [Vani有约会]雨天的尾巴 LG传送门 线段树合并入门好题. 先别急着上线段树合并,考虑一下这题的暴力.一看就是树上差分,对于每一个节点统计每种救济粮的数量,再一遍dfs把差分的结果统计成答案.如果 ...

  5. 【BZOJ3307】雨天的尾巴 线段树合并

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

  6. BZOJ3307雨天的尾巴——线段树合并

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

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

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

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

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

  9. P4556 雨天的尾巴 线段树合并

    使用线段树合并,每个节点维护一棵权值线段树,下标为救济粮种类,区间维护数量最多的救济粮编号(下标).所以每个节点答案即为\(tre[rot[x]]\). 然后运用树上点的差分思想,对于分发路径\(u, ...

随机推荐

  1. (转)Javascript中console.log()用法

    原文地址应该是这个吧:http://my.oschina.net/junn/blog/142728 注意:必须要提前打开IE的开发者模式才能看到输入 否则就会报错. IE下可以这个判断: if (co ...

  2. maven配置文件注意事项

    1:安装完成后在C:\Users\Administrator\.m2有一个文件settings.xml需要修改一下配置 2:设置maven从网上下载的jar包.(时间长会很大).我设置它保存我的电脑位 ...

  3. Linux 常用命令五 软链接和硬链接

    一.软链接 相当于windows的快捷方式,当源文件不存在时,软链接失效. 创建软链接: wang@wang:~/workpalce/python$ ls -l 总用量 -rw-rw-r-- wang ...

  4. bzoj 3824: [Usaco2014 Dec]Guard Mark【状压dp】

    设f[s]为已经从上到下叠了状态为s的牛的最大稳定度,转移的话枚举没有在集合里并且强壮度>=当前集合牛重量和的用min(f[s],当前放进去的牛还能承受多种)来更新,高度的话直接看是否有合法集合 ...

  5. jQuery查找

    导航查找方法: 向下查找兄弟标签: $().next() $().nextAll() 向上查找兄弟标签: 可以查找所有兄弟标签: 查找子标签 查找父级标签: $().parent() $().pare ...

  6. [poj3744] Scout YYF I【概率dp 数学期望】

    传送门:http://poj.org/problem?id=3744 令f(i)表示到i,安全的概率.则f(i) = f(i - 1) * p + f(i - 2) * (1 - p),若i位置有地雷 ...

  7. 递推DP UVA 590 Always on the run

    题目传送门 题意:题意难懂,就是一个小偷在m天内从城市1飞到城市n最小花费,输入的是每个城市飞到其他城市的航班. 分析:dp[i][j] 表示小偷第i天在城市j的最小花费.状态转移方程:dp[i][j ...

  8. VS2010的一个错误,无法加载类型

    一个解决方案中的一个项目X,启动时总是报错,无法加载一个同一个解决方案中另一个项目A生成EXE中的数据类型. 做了如下的步骤解决问题. 1:检查项目A,未发现错误,调试启动A,一切正常. 2:检查项目 ...

  9. go环境搭建及vscode中调试

    1.下载go安装包一般国内用户无法在官网下载,可以自行百度找一些共享的资源墙内下载地址: http://www.golangtc.com/downloadCSDN上资源下载(一般需要积分):http: ...

  10. OC中protocol、category和继承的关系--转

    开放封闭原则(OCP)就是,“对扩展开放,对更改封闭”.是所有面向对象设计的一个核心宗旨.感兴趣的可以看百度百科的一些解释:http://baike.baidu.com/view/2493421.ht ...