在对树中数据进行改动的时候需要很多pushdown(具体操作见代码),不然会wa,大概原因和线段树区间修改需要很多pushup是一样的。
这个轻重链的方法特别好用,虽然第一次写树链剖分但是容易理解又有优秀复杂度的结构让人情不自禁orz。
(后来发现很久以前学lca的时候就学了树链剖分只不过忘了,mdzz)
因为忘了去掉测试代码的freopen,re了4次(虽然有三次就算不re也wa),发现一行一行检查对于数据结构的调试很有用(虽然一直找不出来re原因但是我起码避免了很多wa呀)。
 
代码
 #include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
using namespace std;
const int maxn=;
const long long modn=;
const int minf=<<;
int n,m;char ch[]={};
struct nod{
int y,next,v;
}e[maxn*];
int head[maxn]={},tot=;
struct seg{
int mi,mx,sum,c,l,r;
int f,rev;
seg(){mi=mx=sum=c=l=r=;f=rev=;}
}t[maxn*];
int dep[maxn]={},crs[maxn]={},pos[maxn]={},road[maxn]={};
int kid[maxn]={},fa[maxn]={},top[maxn]={};
void init(int x,int y,int v){
e[++tot].v=v;e[tot].y=y;
e[tot].next=head[x];head[x]=tot;
}
int dfs1(int x,int pa){
int y,siz=,tsn=,ma=;
fa[x]=pa;dep[x]=dep[pa]+;
for(int i=head[x];i;i=e[i].next){
y=e[i].y;
if(y==pa)continue;
siz=dfs1(y,x);tsn+=siz;
if(siz>ma)ma=siz,kid[x]=y;
crs[(i+)/]=y;
}
return tsn;
}
void dfs2(int x,int pa){
int y,v;
pos[x]=++tot;top[x]=pa;
if(kid[x])dfs2(kid[x],pa);
for(int i=head[x];i;i=e[i].next){
y=e[i].y;v=e[i].v;
if(y==kid[x])road[pos[kid[x]]]=v;
if(y==kid[x]||y==fa[x])continue;
road[tot+]=v;
dfs2(y,y);
}
}
void pushup(int x){
int l=x*,r=x*+;
t[x].mx=max(t[l].mx,t[r].mx);
t[x].mi=min(t[l].mi,t[r].mi);
t[x].sum=t[l].sum+t[r].sum;
}
void pushdown(int x){
if(t[x].f){
int l=x*,r=x*+;
t[x].rev=t[x].f=;
t[x].mx=t[x].mi=t[x].c;
t[x].sum=t[x].c*(t[x].r-t[x].l+);
if(t[x].l<t[x].r){
t[l].c=t[r].c=t[x].c;
t[l].f=t[r].f=;
t[l].rev=t[r].rev=;
}
return;
}
if(t[x].rev){
int l=x*,r=x*+;
t[x].rev=;
swap(t[x].mi,t[x].mx);
t[x].mx*=-;t[x].mi*=-;t[x].sum*=-;
if(t[x].l<t[x].r){
t[l].c*=-;t[l].rev^=;
t[r].c*=-;t[r].rev^=;
}
}
}
void build(int x,int l,int r){
t[x].l=l,t[x].r=r;
if(l==r){
t[x].mx=t[x].mi=t[x].sum=road[l];
return;
}
int mid=(l+r)/;
build(x*,l,mid);
build(x*+,mid+,r);
pushup(x);
}
void cov(int x,int l,int r,int w){
if(t[x].l>=l&&t[x].r<=r){
t[x].f=;t[x].c=w; pushdown(x);
return;
}
pushdown(x);
int mid=(t[x].l+t[x].r)/;
if(l<=mid)cov(x*,l,r,w);
pushdown(x*);
if(r>mid) cov(x*+,l,r,w);
pushdown(x*+);
pushup(x);
}
void revs(int x,int l,int r){
if(t[x].l>=l&&t[x].r<=r){
t[x].c*=-;t[x].rev^=;
pushdown(x);
return;
}pushdown(x);
int mid=(t[x].l+t[x].r)/;
if(l<=mid) revs(x*,l,r);
pushdown(x*);
if(r>mid) revs(x*+,l,r);
pushdown(x*+);
pushup(x);
}
int mi(int x,int l,int r){
pushdown(x);
if(t[x].l>=l&&t[x].r<=r){
return t[x].mi;
}
int mid=(t[x].l+t[x].r)/,minn=minf;
if(l<=mid) minn=min(minn,mi(x*,l,r));
if(r>mid) minn=min(minn,mi(x*+,l,r));
return minn;
}
int mx(int x,int l,int r){
pushdown(x);
if(t[x].l>=l&&t[x].r<=r){
return t[x].mx;
}
int mid=(t[x].l+t[x].r)/,mxn=-minf;
if(l<=mid) mxn=max(mxn,mx(x*,l,r));
if(r>mid) mxn=max(mxn,mx(x*+,l,r));
return mxn;
}
int su(int x,int l,int r){
pushdown(x);
if(t[x].l>=l&&t[x].r<=r){
return t[x].sum;
}
int mid=(t[x].l+t[x].r)/,sumn=;
if(l<=mid) sumn+=su(x*,l,r);
if(r>mid) sumn+=su(x*+,l,r);
return sumn;
}
void mrev(int x,int y){
int a,b;
for(a=top[x],b=top[y];a!=b;){
if(dep[a]<dep[b])swap(a,b),swap(x,y);
revs(,pos[a],pos[x]);
x=fa[a];a=top[x];
}
if(dep[x]>dep[y])swap(x,y);
if(x!=y) revs(,pos[x]+,pos[y]);
}
void doit(int x,int y,int k){
int a,b,mxn=-minf,minn=minf,sumn=;
for(a=top[x],b=top[y];a!=b;){
if(dep[a]<dep[b])swap(a,b),swap(x,y);
if(k==) sumn+=su(,pos[a],pos[x]);
else if(k==) mxn=max(mx(,pos[a],pos[x]),mxn);
else minn=min(mi(,pos[a],pos[x]),minn);
x=fa[a];a=top[x];
}
if(dep[x]>dep[y])swap(x,y);
if(x!=y){
if(k==) sumn+=su(,pos[x]+,pos[y]);
else if(k==) mxn=max(mx(,pos[x]+,pos[y]),mxn);
else minn=min(mi(,pos[x]+,pos[y]),minn);
}
if(k==) printf("%d\n",sumn);
else if(k==) printf("%d\n",mxn);
else printf("%d\n",minn);
}
int main(){
//freopen("wtf.in","r",stdin);
//freopen("wtf.out","w",stdout);
scanf("%d",&n);int x,y,z;
for(int i=;i<n;i++){
scanf("%d%d%d",&x,&y,&z);
x++;y++;
init(x,y,z);init(y,x,z);
}tot=;dfs1(,);dfs2(,);
build(,,n);
scanf("%d",&m);
for(int i=;i<=m;i++){
scanf("%s%d%d",&ch,&x,&y);
if(ch[]=='N') mrev(x+,y+);
else if(ch[]=='C')cov(,pos[crs[x]],pos[crs[x]],y);
else if(ch[]=='U')doit(x+,y+,);
else if(ch[]=='A')doit(x+,y+,);
else if(ch[]=='I')doit(x+,y+,);
}
return ;
}

BZOJ2157: 旅游 树链剖分 线段树的更多相关文章

  1. BZOJ2157旅游——树链剖分+线段树

    题目描述 Ray 乐忠于旅游,这次他来到了T 城.T 城是一个水上城市,一共有 N 个景点,有些景点之间会用一座桥连接.为了方便游客到达每个景点但又为了节约成本,T 城的任意两个景点之间有且只有一条路 ...

  2. bzoj 2157: 旅游【树链剖分+线段树】

    裸的树链剖分+线段树 但是要注意一个地方--我WA了好几次才发现取完相反数之后max值和min值是要交换的-- #include<iostream> #include<cstdio& ...

  3. 【BZOJ-2325】道馆之战 树链剖分 + 线段树

    2325: [ZJOI2011]道馆之战 Time Limit: 40 Sec  Memory Limit: 256 MBSubmit: 1153  Solved: 421[Submit][Statu ...

  4. 【BZOJ2243】[SDOI2011]染色 树链剖分+线段树

    [BZOJ2243][SDOI2011]染色 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的 ...

  5. BZOJ2243 (树链剖分+线段树)

    Problem 染色(BZOJ2243) 题目大意 给定一颗树,每个节点上有一种颜色. 要求支持两种操作: 操作1:将a->b上所有点染成一种颜色. 操作2:询问a->b上的颜色段数量. ...

  6. POJ3237 (树链剖分+线段树)

    Problem Tree (POJ3237) 题目大意 给定一颗树,有边权. 要求支持三种操作: 操作一:更改某条边的权值. 操作二:将某条路径上的边权取反. 操作三:询问某条路径上的最大权值. 解题 ...

  7. bzoj4034 (树链剖分+线段树)

    Problem T2 (bzoj4034 HAOI2015) 题目大意 给定一颗树,1为根节点,要求支持三种操作. 操作 1 :把某个节点 x 的点权增加 a . 操作 2 :把某个节点 x 为根的子 ...

  8. HDU4897 (树链剖分+线段树)

    Problem Little Devil I (HDU4897) 题目大意 给定一棵树,每条边的颜色为黑或白,起始时均为白. 支持3种操作: 操作1:将a->b的路径中的所有边的颜色翻转. 操作 ...

  9. Aizu 2450 Do use segment tree 树链剖分+线段树

    Do use segment tree Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://www.bnuoj.com/v3/problem_show ...

  10. 【POJ3237】Tree(树链剖分+线段树)

    Description You are given a tree with N nodes. The tree’s nodes are numbered 1 through N and its edg ...

随机推荐

  1. Spring Session加Redis

    session是一个非常常见的概念.session的作用是为了辅助http协议,因为http是本身是一个无状态协议.为了记录用户的状态,session机制就应运而生了.同时session也是一个非常老 ...

  2. python常用库之base64

    1. 什么是base64 base64是一种将不可见字符转换为可见字符的编码方式. 2. 如何使用 最简单的使用方式 import base64 if __name__ == '__main__': ...

  3. python笔记之BytesIO

    1. 什么是BytesIO BytesIO与StringIO类似,不同的是StringIO只能存放string,BytesIO是用来存放bytes的,它提供了在内存中读写字节的能力. 即在内存中读写字 ...

  4. 爬虫实战--基于requests和beautifulsoup的妹子网图片爬取(福利哦!)

    #coding=utf-8 import requests from bs4 import BeautifulSoup import os all_url = 'http://www.mzitu.co ...

  5. 【译】第一篇 SQL Server代理概述

    本篇文章是SQL Server代理系列的第一篇,详细内容请参考原文. SQL Server代理是SQL Server的作业调度和告警服务,如果使用得当,它可以大大简化DBA的工作量.SQL Serve ...

  6. perl6 Socket: 发送HTTP请求

    sub MAIN(Str $host,Str $path, Int $port) { my $send = "GET $path HTTP/1.1\r\nHost: $host\r\n\r\ ...

  7. Linux kernel kfifo分析【转】

    转自:https://zohead.com/archives/linux-kernel-kfifo/ 本文同步自(如浏览不正常请点击跳转):https://zohead.com/archives/li ...

  8. Mathtype公式位置偏上

    Mathtype公式位置偏上 部分Mathtype公式与文档文字没有很好的对齐,而是浮起来了,也就是说Mathtype公式的位置比正常文字稍高,这是我写论文时碰到的一个很麻烦的问题.然后就是行距稍微大 ...

  9. Gradient-Based Learning Applied to Document Recognition 部分阅读

    卷积网络        卷积网络用三种结构来确保移位.尺度和旋转不变:局部感知野.权值共享和时间或空间降采样.典型的leNet-5如下图所示: C1中每个特征图的每个单元和输入的25个点相连,这个5* ...

  10. MySQL之正则表达式

    一.介绍 正则表达式用来描述或者匹配符合规则的字符串.它的用法和like比较相似,但是它又比like更强大,能够实现一些很特殊的规则匹配:正则表达式需要使用REGEXP命令,匹配上返回"1& ...