[BZOJ 3531] [Sdoi2014] 旅行 【离线+LCT】
题目链接:BZOJ - 3531
题目分析
题目询问一条路径上的信息时,每次询问有某种特定的文化的点。
每个点的文化就相当于一种颜色,每次询问一条路径上某种颜色的点的信息。
可以使用离线算法, 类似于“郁闷的小 J ” 那道题目。将各种操作和询问按照颜色为第一关键字,时间为第二关键字排序。
那么修改颜色的操作就相当于在原颜色中是删点,在新颜色中是加点。
处理完一种颜色的操作后,要将这个颜色的点都做一次删除操作,这样,对于处理下一种颜色,树就又是空的了。
这种题,思考的时候有点晕,写代码的时候非常愉悦。
代码
#include <iostream>
#include <cstdlib>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <queue> using namespace std; inline void Read(int &Num)
{
char c = getchar();
bool Neg = false;
while (c < '0' || c > '9')
{
if (c == '-') Neg = true;
c = getchar();
}
Num = c - '0'; c = getchar();
while (c >= '0' && c <= '9')
{
Num = Num * 10 + c - '0';
c = getchar();
}
if (Neg) Num = -Num;
} inline int gmin(int a, int b) {return a < b ? a : b;}
inline int gmax(int a, int b) {return a > b ? a : b;} const int MaxN = 100000 + 5, MaxM = 100000 + 5, INF = 999999999; int n, m, QTop, ATot;
int W[MaxN], C[MaxN], WT[MaxN], CT[MaxN], Ans[MaxM], Father[MaxN], T[MaxN], Sum[MaxN], Max[MaxN], Son[MaxN][2]; bool isRoot[MaxN], Rev[MaxN]; struct Edge
{
int v;
Edge *Next;
} E[MaxN * 2], *P = E, *Point[MaxN]; inline void AddEdge(int x, int y)
{
++P; P -> v = y;
P -> Next = Point[x]; Point[x] = P;
} struct ES
{
int Type, Col, Pos, Num, x, y, TL, AIdx;
/*
TypeList:
1 : Change City-Pos to Num
2 : Qsum of the path(x, y)
3 : Qmax of the path(x, y)
*/
} Q[MaxN * 2 + MaxM * 2]; inline bool Cmp(ES e1, ES e2)
{
if (e1.Col != e2.Col) return e1.Col < e2.Col;
return e1.TL < e2.TL;
} queue<int> Qe; void BFS()
{
while (!Qe.empty()) Qe.pop();
Qe.push(1); Father[1] = 0;
int x;
while (!Qe.empty())
{
x = Qe.front(); Qe.pop();
for (Edge *j = Point[x]; j; j = j -> Next)
{
if (j -> v == Father[x]) continue;
Father[j -> v] = x;
Qe.push(j -> v);
}
}
} /************************ LCT Start ****************************/ inline void Update(int x)
{
Sum[x] = Sum[Son[x][0]] + Sum[Son[x][1]] + T[x];
Max[x] = gmax(T[x], gmax(Max[Son[x][0]], Max[Son[x][1]]));
} inline void Reverse(int x)
{
Rev[x] = !Rev[x];
swap(Son[x][0], Son[x][1]);
} inline void PushDown(int x)
{
if (!Rev[x]) return;
Rev[x] = false;
if (Son[x][0]) Reverse(Son[x][0]);
if (Son[x][1]) Reverse(Son[x][1]);
} inline int GetDir(int x) {return x == Son[Father[x]][0] ? 0 : 1;} void Rotate(int x)
{
int y = Father[x], f = GetDir(x) ^ 1;
PushDown(y); PushDown(x);
if (isRoot[y])
{
isRoot[y] = false;
isRoot[x] = true;
}
else Son[Father[y]][GetDir(y)] = x;
Father[x] = Father[y];
Son[y][f ^ 1] = Son[x][f];
if (Son[x][f]) Father[Son[x][f]] = y;
Son[x][f] = y; Father[y] = x;
Update(y); Update(x);
} void Splay(int x)
{
int y;
while (!isRoot[x])
{
y = Father[x];
if (isRoot[y])
{
Rotate(x);
break;
}
if (GetDir(x) == GetDir(y)) Rotate(y);
else Rotate(x);
Rotate(x);
}
} inline int Access(int x)
{
int y = 0;
while (x != 0)
{
Splay(x);
PushDown(x);
if (Son[x][1]) isRoot[Son[x][1]] = true;
Son[x][1] = y;
if (y) isRoot[y] = false;
Update(x);
y = x;
x = Father[x];
}
return y;
} inline void Make_Root(int x)
{
int t = Access(x);
Reverse(t);
} /************************ LCT End ****************************/ int main()
{
scanf("%d%d", &n, &m);
for (int i = 1; i <= n; ++i)
{
Read(W[i]); Read(C[i]);
WT[i] = W[i]; CT[i] = C[i];
}
int a, b;
for (int i = 1; i <= n - 1; ++i)
{
Read(a); Read(b);
AddEdge(a, b); AddEdge(b, a);
}
char Str[5];
for (int i = 1; i <= n; ++i)
{
++QTop; Q[QTop].TL = 0; Q[QTop].Col = CT[i];
Q[QTop].Type = 1; Q[QTop].Pos = i; Q[QTop].Num = WT[i];
}
for (int i = 1; i <= m; ++i)
{
scanf("%s", Str);
Read(a); Read(b);
if (strcmp(Str, "CC") == 0)
{
++QTop; Q[QTop].TL = i; Q[QTop].Col = CT[a];
Q[QTop].Type = 1; Q[QTop].Pos = a; Q[QTop].Num = 0;
++QTop; Q[QTop].TL = i; Q[QTop].Col = b;
Q[QTop].Type = 1; Q[QTop].Pos = a; Q[QTop].Num = WT[a];
CT[a] = b;
}
else if (strcmp(Str, "CW") == 0)
{
++QTop; Q[QTop].TL = i; Q[QTop].Col = CT[a];
Q[QTop].Type = 1; Q[QTop].Pos = a; Q[QTop].Num = b;
WT[a] = b;
}
else if (strcmp(Str, "QS") == 0)
{
++QTop; Q[QTop].TL = i; Q[QTop].Col = CT[a];
Q[QTop].Type = 2; Q[QTop].x = a; Q[QTop].y = b; Q[QTop].AIdx = ++ATot;
}
else if (strcmp(Str, "QM") == 0)
{
++QTop; Q[QTop].TL = i; Q[QTop].Col = CT[a];
Q[QTop].Type = 3; Q[QTop].x = a; Q[QTop].y = b; Q[QTop].AIdx = ++ATot;
}
}
for (int i = 1; i <= n; ++i)
{
++QTop; Q[QTop].TL = INF; Q[QTop].Col = CT[i];
Q[QTop].Type = 1; Q[QTop].Pos = i; Q[QTop].Num = 0;
}
sort(Q + 1, Q + QTop + 1, Cmp);
BFS();
for (int i = 1; i <= n; ++i)
{
isRoot[i] = true; Rev[i] = false;
T[i] = 0; Max[i] = 0; Sum[i] = 0;
}
int t;
for (int i = 1; i <= QTop; ++i)
{
if (Q[i].Type == 1)
{
Splay(Q[i].Pos);
T[Q[i].Pos] = Q[i].Num;
Update(Q[i].Pos);
}
else
{
Make_Root(Q[i].x);
t = Access(Q[i].y);
if (Q[i].Type == 2) Ans[Q[i].AIdx] = Sum[t];
else Ans[Q[i].AIdx] = Max[t];
}
}
for (int i = 1; i <= ATot; ++i) printf("%d\n", Ans[i]);
return 0;
}
[BZOJ 3531] [Sdoi2014] 旅行 【离线+LCT】的更多相关文章
- BZOJ 3531: [Sdoi2014]旅行 [树链剖分]
3531: [Sdoi2014]旅行 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 1685 Solved: 751[Submit][Status] ...
- bzoj 3531 [Sdoi2014]旅行(树链剖分,线段树)
3531: [Sdoi2014]旅行 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 876 Solved: 446[Submit][Status][ ...
- bzoj 3531 [Sdoi2014]旅行 (树剖+线段树 动态开点)
3531: [Sdoi2014]旅行 Time Limit: 40 Sec Memory Limit: 512 MBSubmit: 2984 Solved: 1312[Submit][Status ...
- BZOJ 3531 [Sdoi2014]旅行 树链剖分+动态开点线段树
题意 S国有N个城市,编号从1到N.城市间用N-1条双向道路连接,满足从一个城市出发可以到达其它所有城市.每个城市信仰不同的宗教,如飞天面条神教.隐形独角兽教.绝地教都是常见的信仰. 为了方便,我们用 ...
- [bzoj 3531][SDOI2014]旅行(树链剖分+动态开点线段树)
题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=3531 分析: 对于每个颜色(颜色<=10^5)都建立一颗线段树 什么!那么不是M ...
- bzoj 3531: [Sdoi2014]旅行
Description S国有N个城市,编号从1到N.城市间用N-1条双向道路连接,满足从一个城市出发可以到达其它所有城市.每个城市信仰不同的宗教,如飞天面条神教.隐形独角兽教.绝地教都是常见的信仰. ...
- 【刷题】BZOJ 3531 [Sdoi2014]旅行
Description S国有N个城市,编号从1到N.城市间用N-1条双向道路连接,满足 从一个城市出发可以到达其它所有城市.每个城市信仰不同的宗教,如飞天面条神教.隐形独角兽教.绝地教都是常见的信仰 ...
- BZOJ 3531 SDOI2014 旅行 树链剖分+线段树动态开点
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=3531 题意概述: 给出一棵N个点的树,树上的每个结点有一个颜色和权值,支持以下四种操作: ...
- BZOJ 3531: [Sdoi2014]旅行 权值线段树 + 树链剖分
Description S国有N个城市,编号从1到N.城市间用N-1条双向道路连接,满足 从一个城市出发可以到达其它所有城市.每个城市信仰不同的宗教,如飞天面条神教.隐形独角兽教.绝地教都是常见的信仰 ...
随机推荐
- 使用Keytool工具生成证书Keystore和证书签名请求文件
内容概览: keytool的几个常用的命令. 1.创建证书 2.查看证书库 3.导出证书文件 4.导入证书的信息 5.查看证书信息 6.删除密钥库中的条目 7.修改证书条目的口令 ---------- ...
- C#扫盲之:前台线程后台线程
1.线程分类 线程由程序员创建,可是创建的方式不同,总体来说有两种,一种是个人构造,也就是使用thread类new线程对象创建,这一类线程是大部分程序员知道的,也叫专用线程;还有一种是由CLR创建,这 ...
- 【MINA】字节序知识
字节序,分为高位在前和低位在前,说白了就是先从低操作还是从高位操作 java和网络的字节序是一致的,都是高位在前,这意味着java端序列化和反序列化时不用关心字节序的问题, 那问题是,那讨论字节序有什 ...
- LSJ_NHibernate第三章 IDAL,DAL,BLL
前言: 做项目(面向数据编程),首先必须了解业务,这是核心,不懂业务写出来的代码毫无意义.业务我这里分为两种,简单业务操作,复杂业务操作,我以他们操作表的界限进行区分,假设我更新一条数据,只操作了一张 ...
- sdk 提示至少需要Build-tools 19.1.0以上的解决方式
网上搜了很多办法,例如修改host文件或者设置sdk manager tool 为force方式都解决不了,最简单的方式是手动下载build-tools下的包,例如直接百度搜索build-tools ...
- SqlServer Change Data Capture(CDC)数据变更捕获
最近在使用SqlServer2008r2数据库做系统的时候,在某些重要的.经常涉及到修改的表上,想加上一些恢复机制,一开始想找找看看有没有类似Oracle数据库闪回那样的功能,后来发现CDC的功能可以 ...
- php注册登录时生成的验证码
http://blog.sina.com.cn/s/blog_8173443e01012l82.html 记得我学php时第一件事就是研究登陆注册.当然,登陆少不了验证码.两年过去了,昨天突然想用个验 ...
- Oracle 11g之创建和管理表练习
创建表: SQL> create table name (empno number(4), ename VARCHAR2(10)); 查看表结构: desc name; SQL> desc ...
- iOS 计算两个日期之间的天数问题
//获取当前时间若干年.月.日之后的时间 + (NSDate *)dateWithFromDate:(NSDate *)date years:(NSInteger)years months:(NSIn ...
- sqlite3 语句总结
一. iOS客户端设计数据库时一般使用 sqlite,以sqlite3 为例,简单介绍一下. 二. sqlite3常用命令当前目录下建立或打开test.db数据库文件,并进入sqlite命令终端 ...