https://www.lydsy.com/JudgeOnline/problem.php?id=3531

首先这题意要求树链上的最大值以及求和,其树链剖分的做法已经昭然若揭

问题在于这次的信息有宗教条件下的限制,导致不那么容易维护。

第一个想法自然是对于每一个宗教都建一颗线段树,看一下数据范围,宗教的范围是1e5,N的范围也是1e5,又好像空间不那么允许。

事实上可以采用动态线段树的方法节省一波空间,整个做法就变得科学了起来。

如果不是因为我的愚蠢在树剖的重链部分写挂了导致T了很久,这题还是很温暖的

#include <map>
#include <set>
#include <ctime>
#include <cmath>
#include <queue>
#include <stack>
#include <vector>
#include <string>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <sstream>
#include <iostream>
#include <algorithm>
#include <functional>
using namespace std;
#define For(i, x, y) for(int i=x;i<=y;i++)
#define _For(i, x, y) for(int i=x;i>=y;i--)
#define Mem(f, x) memset(f,x,sizeof(f))
#define Sca(x) scanf("%d", &x)
#define Sca2(x,y) scanf("%d%d",&x,&y)
#define Sca3(x,y,z) scanf("%d%d%d",&x,&y,&z)
#define Scl(x) scanf("%lld",&x);
#define Pri(x) printf("%d\n", x)
#define Prl(x) printf("%lld\n",x);
#define CLR(u) for(int i=0;i<=N;i++)u[i].clear();
#define LL long long
#define ULL unsigned long long
#define mp make_pair
#define PII pair<int,int>
#define PIL pair<int,long long>
#define PLL pair<long long,long long>
#define pb push_back
#define fi first
#define se second
typedef vector<int> VI;
int read(){int x = ,f = ;char c = getchar();while (c<'' || c>''){if (c == '-') f = -;c = getchar();}
while (c >= ''&&c <= ''){x = x * + c - '';c = getchar();}return x*f;}
const double eps = 1e-;
const int maxn = 1e5 + ;
const int INF = 0x3f3f3f3f;
const int mod = 1e9 + ;
int N,M,K;
inline int max(int a,int b){
return a>b?a:b;
}
PII node[maxn];
struct Edge{
int to,next;
}edge[maxn * ];
int head[maxn],Tot;
void init(){
for(int i = ; i <= N ; i ++) head[i] = -;
Tot = ;
}
void add(int u,int v){
edge[Tot].to = v;
edge[Tot].next = head[u];
head[u] = Tot++;
}
int size[maxn],top[maxn],fa[maxn],son[maxn];
int dep[maxn],pos[maxn];
void dfs(int t,int la){
size[t] = ; son[t] = ;
int heavy = ;
for(int i = head[t]; ~i ; i = edge[i].next){
int v = edge[i].to;
if(v == la) continue;
fa[v] = t;
dep[v] = dep[t] + ;
dfs(v,t);
if(heavy < size[v]){
heavy = size[v];
son[t] = v;
}
size[t] += size[v];
}
}
int cnt;
void dfs2(int t,int la){
top[t] = la;
pos[t] = ++cnt;
if(!son[t]) return;
dfs2(son[t],la);
for(int i = head[t]; ~i ; i = edge[i].next){
int v = edge[i].to;
if(fa[t] == v || son[t] == v) continue;
dfs2(v,v);
}
}
struct Tree{
int Max,sum;
int lt,rt;
void init(){
Max = sum = lt = rt = ;
}
}tree[maxn * ];
int tot;
int thead[maxn];
void check(int &t){
if(t) return;
t = ++tot;
tree[t].init();
}
void Pushup(int t){
int ls = tree[t].lt,rs = tree[t].rt;
check(ls);check(rs);
tree[t].sum = tree[ls].sum + tree[rs].sum;
tree[t].Max = max(tree[ls].Max,tree[rs].Max);
}
void update(int &t,int l,int r,int p,int w){
check(t);
if(l == r){
tree[t].Max = tree[t].sum = w;
return;
}
int m = (l + r) >> ;
if(p <= m) update(tree[t].lt,l,m,p,w);
else update(tree[t].rt,m + ,r,p,w);
Pushup(t);
}
void update(int i,int w){
update(thead[node[i].se],,N,pos[i],w);
}
int query(int &t,int l,int r,int L,int R,int p){
check(t);
if(L <= l && r <= R){
if(p) return tree[t].sum;
else return tree[t].Max;
}
int m = (l + r) >> ;
if(R <= m) return query(tree[t].lt,l,m,L,R,p);
else if(L > m) return query(tree[t].rt,m + ,r,L,R,p);
else{
if(p) return query(tree[t].lt,l,m,L,m,p) + query(tree[t].rt,m + ,r,m + ,R,p);
else return max(query(tree[t].lt,l,m,L,m,p),query(tree[t].rt,m + ,r,m + ,R,p));
}
}
int query(int u,int v,int flag){
int ans = ;
int &c = thead[node[u].se];
while(top[u] != top[v]){
if(dep[top[u]] < dep[top[v]]) swap(u,v);
if(flag) ans += query(c,,N,pos[top[u]],pos[u],);
else ans = max(ans,query(c,,N,pos[top[u]],pos[u],));
u = fa[top[u]];
}
if(dep[u] > dep[v]) swap(u,v);
if(flag) ans += query(c,,N,pos[u],pos[v],);
else ans = max(ans,query(c,,N,pos[u],pos[v],));
return ans;
}
int main(){
Sca2(N,M); init();
for(int i = ; i <= N ; i ++) node[i].fi = read(),node[i].se = read();
for(int i = ; i <= N - ; i ++){
int u = read(),v = read();
add(u,v); add(v,u);
}
int root = ;
dfs(root,); dfs2(root,root);
for(int i = ; i <= N; i ++) update(i,node[i].fi);
for(int i = ; i <= M ; i ++){
char op[];
scanf("%s",op);
int x = read(),y = read();
if(op[] == 'C'){
if(op[] == 'C'){
update(x,);
node[x].se = y;
update(x,node[x].fi);
}else{
node[x].fi = y;
update(x,y);
}
}else{
if(op[] == 'S'){
Pri(query(x,y,)); //sum
}else{
Pri(query(x,y,)); //Max
}
}
}
return ;
}

BZOJ3531 树剖 + 动态开点线段树的更多相关文章

  1. BZOJ 3531: [Sdoi2014]旅行 (树剖+动态开点线段树)

    对于每种信仰维护一棵动态开点线段树就行了- #include <cstdio> #include <cctype> #include <cstring> #incl ...

  2. CF915E Physical Education Lessons 动态开点线段树

    题目链接 CF915E Physical Education Lessons 题解 动态开点线段树 代码 /* 动态开点线段树 */ #include<cstdio> #include&l ...

  3. bzoj3531: [Sdoi2014]旅行 (树链剖分 && 动态开点线段树)

    感觉动态开点线段树空间复杂度好优秀呀 树剖裸题 把每个宗教都开一颗线段树就可以了 但是我一直TLE 然后调了一个小时 为什么呢 因为我 #define max(x, y) (x > y ? x ...

  4. NFLSOJ #917 -「lych_cys模拟题2018」橘子树(树剖+ODT+莫反统计贡献的思想+动态开点线段树)

    题面传送门 sb 出题人不在题面里写 \(b_i=0\) 导致我挂成零蛋/fn/fn 首先考虑树链剖分将路径问题转化为序列上的问题,因此下文中简称"位置 \(i\)"表示 DFS ...

  5. [ZJOI2019]语言(树链剖分+动态开点线段树+启发式合并)

    首先,对于从每个点出发的路径,答案一定是过这个点的路径所覆盖的点数.然后可以做树上差分,对每个点记录路径产生总贡献,然后做一个树剖维护,对每个点维护一个动态开点线段树.最后再从根节点开始做一遍dfs, ...

  6. [2016湖南长沙培训Day4][前鬼后鬼的守护 chen] (动态开点线段树+中位数 or 动规 or 贪心+堆优化)

    题目大意 给定一个长度为n的正整数序列,令修改一个数的代价为修改前后两个数的绝对值之差,求用最小代价将序列转换为不减序列. 其中,n满足小于500000,序列中的正整数小于10^9 题解(引自mzx神 ...

  7. [bzoj 3531][SDOI2014]旅行(树链剖分+动态开点线段树)

    题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=3531 分析: 对于每个颜色(颜色<=10^5)都建立一颗线段树 什么!那么不是M ...

  8. 【BZOJ-4636】蒟蒻的数列 动态开点线段树 ||(离散化) + 标记永久化

    4636: 蒟蒻的数列 Time Limit: 30 Sec  Memory Limit: 256 MBSubmit: 247  Solved: 113[Submit][Status][Discuss ...

  9. codeforces 893F - Physical Education Lessons 动态开点线段树合并

    https://codeforces.com/contest/893/problem/F 题意: 给一个有根树, 多次查询,每次查询对于$x$i点的子树中,距离$x$小于等于$k$的所有点中权值最小的 ...

随机推荐

  1. Process 模块的方法

    join from multiprocessing import Process import time, os def task(name): print('%s is running' % nam ...

  2. codeforces285C

    Building Permutation CodeForces - 285C Permutation p is an ordered set of integers p1,  p2,  ...,  p ...

  3. LOJ2251 [ZJOI2017] 树状数组【线段树】【树套树】

    题目分析: 对于一个$add$操作,它的特点是与树状数组的查询相同,会给$1$到它自己产生影响,而$query$操作则会途径所有包含它的树状数组点.现在$add$操作具有前向性(不会影响之后的点).所 ...

  4. 安卓Android基础第三天——数据库,ListView

    数据库介绍sqlite问:什么情况下使用数据库?答:有大量相似结构的数据需要存储的时候 数据库的创建定义一个类继承SqliteOpenHelpercontext:上下文name:数据库名字,如&quo ...

  5. 2010 SD - ICPC D - Emergency

    D - Emergency Time Limit:1000MS     Memory Limit:65536KB     64bit IO Format:%lld & %llu Submit  ...

  6. 【UOJ349】【WC2018】即时战略 LCT 动态点分治

    这是一道交互题 题目大意 有一棵\(n\)个点的树.最开始\(1\)号点是白的,其他点是黑的. 每次你可以执行一个操作:\(explore(x,y)\).要求\(x\)是一个白点.该函数会返回从\(x ...

  7. MT【260】单调函数

    设$f(x)$是定义在$(0,+\infty)$上的单调函数,且对定义域内的任意实数$x$,都有$f(f(x)-\log_2 x)=3$, 求$f(x)-f^{'}(x)=2$的解所在的区间.____ ...

  8. MS-DOS 6.22 +Vim+masm 汇编环境

    安装vim 个人习惯用 vim 编辑,因此稍微折腾了一下.不用这么麻烦直接用 edit 编辑也是可以的. 原来安装的 MS-DOS 7.10 虚拟机安装好vim后无法运行,所以改用了 MS-DOS 6 ...

  9. 【BZOJ3613】[HEOI2014]南园满地堆轻絮(贪心)

    [BZOJ3613][HEOI2014]南园满地堆轻絮(贪心) 题面 BZOJ 洛谷 题解 考虑二分的做法,每次二分一个答案,那么就会让所有的值尽可能的减少,那么\(O(n)\)扫一遍就好了. 考虑如 ...

  10. Atlantis HDU - 1542 (扫描线,线段树)

    扫描线的模板题,先把信息接收,然后排序,记录下上边和下边,然后用一条虚拟的线从下往上扫.如果我扫到的是下边,那么久用线段树在这个区间内加上1,表示这个区间现在是有的,等我扫描到上边的时候在加上-1,把 ...