1984: 月下“毛景树”

Time Limit: 20 Sec  Memory Limit: 64 MB
Submit: 1728  Solved: 531
[Submit][Status][Discuss]

Description

毛毛虫经过及时的变形,最终逃过的一劫,离开了菜妈的菜园。 毛毛虫经过千山万水,历尽千辛万苦,最后来到了小小的绍兴一中的校园里。爬啊爬~爬啊爬~~毛毛虫爬到了一颗小小的“毛景树”下面,发现树上长着他最爱吃的毛毛果~~~ “毛景树”上有N个节点和N-1条树枝,但节点上是没有毛毛果的,毛毛果都是长在树枝上的。但是这棵“毛景树”有着神奇的魔力,他能改变树枝上毛毛果的个数:  Change k w:将第k条树枝上毛毛果的个数改变为w个。  Cover u v w:将节点u与节点v之间的树枝上毛毛果的个数都改变为w个。  Add u v w:将节点u与节点v之间的树枝上毛毛果的个数都增加w个。 由于毛毛虫很贪,于是他会有如下询问:  Max u v:询问节点u与节点v之间树枝上毛毛果个数最多有多少个。

Input

第一行一个正整数N。 接下来N-1行,每行三个正整数Ui,Vi和Wi,第i+1行描述第i条树枝。表示第i条树枝连接节点Ui和节点Vi,树枝上有Wi个毛毛果。 接下来是操作和询问,以“Stop”结束。

Output

对于毛毛虫的每个询问操作,输出一个答案。

Sample Input

4
1 2 8
1 3 7
3 4 9
Max 2 4
Cover 2 4 5
Add 1 4 10
Change 1 16
Max 2 4
Stop

Sample Output

9
16

【Data Range】
1<=N<=100,000,操作+询问数目不超过100,000。
保证在任意时刻,所有树枝上毛毛果的个数都不会超过10^9个。


比较明显树链剖分
问题边权怎么处理?
转化成边下面的点的权值就行了
 
因为修改操作是第几条边,所以保存mp点到边
 
线段树set修改标记,add增加标记
打set时清空add,打add时有set直接加set就行了
 
注意:不要直接用书上的点的编号的权值作为线段树编号的权值,加一个fid[i]为线段树节点i在原图上的编号!!!!
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long ll;
#define lc o<<1
#define rc o<<1|1
#define m ((l+r)>>1)
#define lson o<<1,l,m
#define rson o<<1|1,m+1,r
const int N=2e5+,INF=1e9+;
int read(){
char c=getchar();int x=,f=;
while(c<''||c>''){if(c=='-')f=-; c=getchar();}
while(c>=''&&c<=''){x=x*+c-''; c=getchar();}
return x*f;
}
int n,u,v,ww,w[N],l,r,mp[N];//edge-->point
char s[];
struct edge{
int v,w,ne,id;
}e[N<<];
int h[N],cnt;
inline void ins(int u,int v,int w,int id){
cnt++;
e[cnt].v=v;e[cnt].w=w;e[cnt].id=id;e[cnt].ne=h[u];h[u]=cnt;
cnt++;
e[cnt].v=u;e[cnt].w=w;e[cnt].id=id;e[cnt].ne=h[v];h[v]=cnt;
}
int tid[N],fa[N],top[N],tot,deep[N],mx[N],size[N],fid[N];
void dfs(int u){
size[u]=;
for(int i=h[u];i;i=e[i].ne){
int v=e[i].v;
if(v==fa[u]) continue;
w[v]=e[i].w;mp[e[i].id]=v;
fa[v]=u;deep[v]=deep[u]+;
dfs(v);
size[u]+=size[v];
if(size[mx[u]]<size[v]) mx[u]=v;
}
}
void dfs(int u,int anc){
if(!u) return;
tid[u]=++tot;fid[tot]=u;
top[u]=anc;
dfs(mx[u],anc);
for(int i=h[u];i;i=e[i].ne){
int v=e[i].v;
if(v!=fa[u]&&v!=mx[u]) dfs(v,v);
}
} struct node{
int mx,set,add;
node():set(-),add(){}
}t[N<<];
inline void merge(int o){
t[o].mx=max(t[lc].mx,t[rc].mx);
}
inline void paintset(int o,int v){
t[o].set=t[o].mx=v;
t[o].add=;
}
inline void paintadd(int o,int v){
if(t[o].set!=-) t[o].set+=v,t[o].mx+=v;
else t[o].add+=v,t[o].mx+=v;
}
inline void pushDown(int o){
if(t[o].set!=-){
paintset(lc,t[o].set);
paintset(rc,t[o].set);
t[o].set=-;
}
if(t[o].add){
paintadd(lc,t[o].add);
paintadd(rc,t[o].add);
t[o].add=;
}
}
void build(int o,int l,int r){
if(l==r) t[o].mx=w[fid[l]];
else{
build(lson);
build(rson);
merge(o);
}
}
void segcha(int o,int l,int r,int p,int v){
if(l==r) t[o].mx=v;
else{
pushDown(o);
if(p<=m) segcha(lson,p,v);
else segcha(rson,p,v);
merge(o);
}
}
void segcov(int o,int l,int r,int ql,int qr,int v){
if(ql<=l&&r<=qr) paintset(o,v);
else{
pushDown(o);
if(ql<=m) segcov(lson,ql,qr,v);
if(m<qr) segcov(rson,ql,qr,v);
merge(o);
}
}
void segadd(int o,int l,int r,int ql,int qr,int v){
if(ql<=l&&r<=qr) paintadd(o,v);
else{
pushDown(o);
if(ql<=m) segadd(lson,ql,qr,v);
if(m<qr) segadd(rson,ql,qr,v);
merge(o);
}
}
int segmx(int o,int l,int r,int ql,int qr){
if(ql<=l&&r<=qr) return t[o].mx;
else{
pushDown(o);
int mx=-INF;
if(ql<=m) mx=max(mx,segmx(lson,ql,qr));
if(m<qr) mx=max(mx,segmx(rson,ql,qr));
return mx;
}
} void add(int x,int y,int v){
while(top[x]!=top[y]){
if(deep[top[x]]<deep[top[y]]) swap(x,y);
segadd(,,n,tid[top[x]],tid[x],v);
x=fa[top[x]];
} if(tid[x]>tid[y]) swap(x,y);
if(x!=y) segadd(,,n,tid[x]+,tid[y],v);//bian quan
}
void cover(int x,int y,int v){
while(top[x]!=top[y]){
if(deep[top[x]]<deep[top[y]]) swap(x,y);
segcov(,,n,tid[top[x]],tid[x],v);
x=fa[top[x]];
} if(tid[x]>tid[y]) swap(x,y);
if(x!=y) segcov(,,n,tid[x]+,tid[y],v);//bian quan
}
int query(int x,int y){
int mx=-INF;
while(top[x]!=top[y]){
if(deep[top[x]]<deep[top[y]]) swap(x,y);
mx=max(mx,segmx(,,n,tid[top[x]],tid[x]));
x=fa[top[x]];
}
if(tid[x]>tid[y]) swap(x,y);
if(x!=y) mx=max(mx,segmx(,,n,tid[x]+,tid[y]));
return mx;
}
int main(){
n=read();
for(int i=;i<=n-;i++) u=read(),v=read(),ww=read(),ins(u,v,ww,i);
dfs();dfs(,);
build(,,n);
while(true){
scanf("%s",s);
if(s[]=='S') break;
else if(s[]=='h') l=read(),v=read(),segcha(,,n,tid[mp[l]],v);
else if(s[]=='o') l=read(),r=read(),v=read(),cover(l,r,v);
else if(s[]=='d') l=read(),r=read(),v=read(),add(l,r,v);
else l=read(),r=read(),printf("%d\n",query(l,r));
}
}
 

BZOJ 1984: 月下“毛景树” [树链剖分 边权]的更多相关文章

  1. Bzoj 1984: 月下“毛景树” 树链剖分

    1984: 月下“毛景树” Time Limit: 20 Sec  Memory Limit: 64 MBSubmit: 1282  Solved: 410[Submit][Status][Discu ...

  2. BZOJ 1984月下“毛景树” LCT维护边权 + 下传标记

    Description 毛毛虫经过及时的变形,最终逃过的一劫,离开了菜妈的菜园. 毛毛虫经过千山万水,历尽千辛万苦,最后来到了小小的绍兴一中的校园里.爬啊爬~爬啊爬~~毛毛虫爬到了一颗小小的“毛景树” ...

  3. BZOJ 1984: 月下“毛景树” (树链剖分+线段树)

    注意赋值和加法的标记下传优先级.具体看代码. CODE #include <vector> #include <queue> #include <cstdio> # ...

  4. BZOJ 1984 月下“毛景树”

    我觉得我要把BZOJ上的链剖写完了吧.... #include<iostream> #include<cstdio> #include<cstring> #incl ...

  5. 【BZOJ】1984 月下“毛景树”

    [算法]树链剖分+线段树 [题解]线段树的区间加值和区间覆盖操作不能同时存在,只能存在一个. 修改:从根节点跑到目标区域路上的标记全部下传,打完标记再上传回根节点(有变动才需要上传). 询问:访问到目 ...

  6. 【BZOJ-1984】月下“毛景树” 树链剖分

    1984: 月下“毛景树” Time Limit: 20 Sec  Memory Limit: 64 MBSubmit: 1314  Solved: 416[Submit][Status][Discu ...

  7. BZOJ1984: 月下“毛景树”

    1984: 月下“毛景树” Time Limit: 20 Sec  Memory Limit: 64 MBSubmit: 713  Solved: 245[Submit][Status] Descri ...

  8. P4315 月下“毛景树”(树链剖分)

    P4315 月下"毛景树"(树链剖分) 题面 简述: 边权转点权(在dfs1处转换) 把一条边权赋值在深度更深的上 需要实现对单边权的染色 , 路径边权的染色 , 路径边权的增加 ...

  9. 【BZOJ1984】月下“毛景树” 树链剖分+线段树

    [BZOJ1984]月下"毛景树" Description 毛毛虫经过及时的变形,最终逃过的一劫,离开了菜妈的菜园. 毛毛虫经过千山万水,历尽千辛万苦,最后来到了小小的绍兴一中的校 ...

随机推荐

  1. 构建ASP.NET MVC4+EF5+EasyUI+Unity2.x注入的后台管理系统(11)-系统日志和异常的处理①

    系列目录 系统需要越来越自动化,我们需要引入日志记录和异常捕获管理员的操作记录需要被记录,看出哪些模块是频繁操作,分析哪些是不必要的功能,哪些是需要被优化的.系统的异常需要被捕获,而不是将系统出错显示 ...

  2. Hawk 4.2 过滤器

    过滤器可以在流中,过滤掉不符合条件的文档.当然也可勾选反向,此时只会留下不符合条件的文档. 空对象过滤器 最为常用,需要列名,可以过滤掉所有内容为Null,或字符串全部都是空字符的情况 数值范围过滤 ...

  3. RPC远程过程调用学习之路(一):用最原始代码还原PRC框架

    RPC: Remote Procedure Call 远程过程调用,即业务的具体实现不是在自己系统中,需要从其他系统中进行调用实现,所以在系统间进行数据交互时经常使用. rpc的实现方式有很多,可以通 ...

  4. SharePoint2013 Set a custom application page as site welcome page

    本文主要介绍如何添加一个custom application page as site welcome page 1.首先创建一个sharepoint 2013 empty solution, add ...

  5. 简单动态规划-LeetCode198

    题目:House Robber You are a professional robber planning to rob houses along a street. Each house has ...

  6. Python(四)装饰器、迭代器&生成器、re正则表达式、字符串格式化

    本章内容: 装饰器 迭代器 & 生成器 re 正则表达式 字符串格式化 装饰器 装饰器是一个很著名的设计模式,经常被用于有切面需求的场景,较为经典的有插入日志.性能测试.事务处理等.装饰器是解 ...

  7. ASP.NET Core 中文文档 第三章 原理(15)请求功能

    作者:Steve Smith 翻译:谢炀(kiler398) 校对:姚阿勇(Dr.Yao).孟帅洋(书缘) 涉及到如何处理 HTTP 请求以及响应的独立 Web 服务器功能已经被分解成独立的接口,这些 ...

  8. passport源码研究

            passport的验证过程主要依赖具体的验证策略来实现的,比较常用的有session策略.local策略和github策略等,验证逻辑都是在这些策略类中定义的.passport模块的定 ...

  9. 前端开发:css技巧,如何设置select、radio 、 checkbox 、file这些不可直接设置的样式 。

    前言: 都说程序员有三宝:人傻,钱多,死得早.博主身边的程序“猿”一大半应了这三宝,这从侧面说明了一个问题,只有理性是过不好日子的.朋友们应该把工作与生活分开,让生活变得感性,让工作变得理性,两者相提 ...

  10. 信贷业务(Ali)

    1.信贷业务视角 信贷业务主要有两个视角,借款人和出资机构.借款人关心借多少钱,还多少钱,多少利息:机构关心信贷资产风险,收益. 领域模型上两个视角分开:个人--->账单.机构--->资产 ...