[洛谷]P1505 [国家集训队]旅游
题目链接:
题目分析:
树剖板,支持单点修改,区间取反,区间求最大值/最小值/和
区间取反取两次等于没取,维护一个\(rev\ tag\),每次打标记用\(xor\)打,记录是否需要翻转,\(push\_down\)里判一下如果要取反就\(-=2 * sum(p)\),容易发现新最大值是原最小值的相反数,最小值同理
细节挺多的,看代码
代码:
#include <bits/stdc++.h>
#define N 2 * (200000 + 5)
#define int long long
#define INF (1000000000 + 7)
using namespace std;
inline int read() {
int cnt = 0, f = 1; char c = getchar();
while (!isdigit(c)) {if (c == '-') f = -f; c = getchar();}
while (isdigit(c)) {cnt = (cnt << 3) + (cnt << 1) + c - '0'; c = getchar();}
return cnt * f;
}
int nxt[N], first[N], to[N], w[N], tot;
void Add(int x, int y, int z) {
nxt[++tot] = first[x];
first[x] = tot;
to[tot] = y;
w[tot] = z;
}
int son[N], dep[N], top[N], siz[N], father[N], id[N], num[N], idx, a[N];
void dfs1(int cur, int fa) {
father[cur] = fa, siz[cur] = 1, dep[cur] = dep[fa] + 1;
for (register int i = first[cur]; i; i = nxt[i]) {
int v = to[i];
if (v != fa) {
a[v] = w[i];
dfs1(v, cur);
siz[cur] += siz[v];
if (siz[son[cur]] < siz[v]) son[cur] = v;
}
}
}
void dfs2(int cur, int tp) {
top[cur] = tp, num[cur] = ++idx, id[idx] = cur;
if (son[cur]) dfs2(son[cur], tp);
for (register int i = first[cur]; i; i = nxt[i]) {
int v = to[i];
if (!num[v]) dfs2(v, v);
}
}
struct node {
int l, r, Min, Max, sum, add;
bool rev;
#define l(p) tree[p].l
#define r(p) tree[p].r
#define Min(p) tree[p].Min
#define Max(p) tree[p].Max
#define sum(p) tree[p].sum
#define rev(p) tree[p].rev
}tree[N << 2];
void push_up(int p) {
sum(p) = sum(p << 1) + sum(p << 1 | 1);
Max(p) = max(Max(p << 1), Max(p << 1 | 1));
Min(p) = min(Min(p << 1), Min(p << 1 | 1));
}
void push_up_rev(int p) {
int gmax = Max(p), gmin = Min(p);
rev(p) ^= 1, sum(p) = -sum(p), Max(p) = -gmin, Min(p) = -gmax;
}
void push_down(int p) {
if (rev(p)) {push_up_rev(p << 1), push_up_rev(p << 1 | 1), rev(p) ^= 1;}
}
void build_tree(int p, int l, int r) {
l(p) = l, r(p) = r;
if (l == r) {
sum(p) = Max(p) = Min(p) = a[id[l]];
return;
}
int mid = (l + r) >> 1;
build_tree(p << 1, l, mid);
build_tree(p << 1 | 1, mid + 1, r);
push_up(p);
}
void modify(int p, int x, int d) {
if (l(p) == r(p)) {
rev(p) = 0, sum(p) = d, Max(p) = d, Min(p) = d;
return;
}
push_down(p);
int mid = (l(p) + r(p)) >> 1;
if (x <= mid) modify(p << 1, x, d);
else modify(p << 1 | 1, x, d);
push_up(p);
}
void Reverse(int p, int l, int r) {
if (l <= l(p) && r >= r(p)) {push_up_rev(p); return;}
push_down(p);
int mid = (l(p) + r(p)) >> 1;
if (l <= mid) Reverse(p << 1, l, r);
if (r > mid) Reverse(p << 1 | 1, l, r);
push_up(p);
}
int query_sum(int p, int l, int r) {
if (l <= l(p) && r >= r(p)) return sum(p);
push_down(p);
int mid = (l(p) + r(p)) >> 1;
long long val = 0;
if (l <= mid) val += query_sum(p << 1, l, r);
if (r > mid) val += query_sum(p << 1 | 1, l, r);
return val;
}
int query_max(int p, int l, int r) {
if (l <= l(p) && r >= r(p)) return Max(p);
push_down(p);
int mid = (l(p) + r(p)) >> 1;
long long val = -INF;
if (l <= mid) val = max(val, query_max(p << 1, l, r));
if (r > mid) val = max(val, query_max(p << 1 | 1, l, r));
return val;
}
int query_min(int p, int l, int r) {
if (l <= l(p) && r >= r(p)) return Min(p);
push_down(p);
int mid = (l(p) + r(p)) >> 1;
long long val = INF;
if (l <= mid) val = min(val, query_min(p << 1, l, r));
if (r > mid) val = min(val, query_min(p << 1 | 1, l, r));
return val;
}
void REVERSE (int u, int v) {
while (top[u] != top[v]) {
if (dep[top[u]] < dep[top[v]]) swap(u, v);
Reverse(1, num[top[u]], num[u]);
u = father[top[u]];
}
if (dep[u] < dep[v]) swap(u, v);
Reverse(1, num[v] + 1, num[u]);
}
int Query_Sum (int u, int v) {
long long val = 0;
while (top[u] != top[v]) {
if (dep[top[u]] < dep[top[v]]) swap(u, v);
val += query_sum(1, num[top[u]], num[u]);
u = father[top[u]];
}
if (dep[u] < dep[v]) swap(u, v);
val += query_sum(1, num[v] + 1, num[u]);
return val;
}
int Query_Max (int u, int v) {
long long val = -INF;
while (top[u] != top[v]) {
if (dep[top[u]] < dep[top[v]]) swap(u, v);
val = max(val, query_max(1, num[top[u]], num[u]));
u = father[top[u]];
}
if (dep[u] < dep[v]) swap(u, v);
val = max(val, query_max(1, num[v] + 1, num[u]));
return val;
}
int Query_Min (int u, int v) {
long long val = INF;
while (top[u] != top[v]) {
if (dep[top[u]] < dep[top[v]]) swap(u, v);
val = min(val, query_min(1, num[top[u]], num[u]));
u = father[top[u]];
}
if (dep[u] < dep[v]) swap(u, v);
val = min(val, query_min(1, num[v] + 1, num[u]));
return val;
}
int n, m, u, v, x, y, z;
char ope[10];
void solve() {
n = read();
for (register int i = 1; i < n; i++) {
x = read(); y = read(); z = read();
Add(x + 1, y + 1, z); Add(y + 1, x + 1, z);
}
dfs1(1, 0); dfs2(1, 1); build_tree(1, 1, n);
m = read();
for (register int i = 1; i <= m; i++) {
scanf("%s", ope + 1);
x = read(); y = read();
if (ope[1] == 'C') {
u = to[2 * x - 1], v = to[2 * x];
if (dep[u] < dep[v]) swap(u, v);
modify(1, num[u], y);
}
if (ope[1] == 'N') {
REVERSE (x + 1, y + 1);
}
if (ope[1] == 'S') {
printf("%lld\n", Query_Sum(x + 1, y + 1));
}
if (ope[1] == 'M') {
if (ope[2] == 'A') printf("%lld\n", Query_Max(x + 1, y + 1));
else printf("%lld\n", Query_Min(x + 1, y + 1));
}
}
}
signed main() {
// freopen("1.in","r",stdin);
solve();
return 0;
}
[洛谷]P1505 [国家集训队]旅游的更多相关文章
- 洛谷 P1505 [国家集训队]旅游 解题报告
P1505 [国家集训队]旅游 题目描述 \(\tt{Ray}\) 乐忠于旅游,这次他来到了\(T\)城.\(T\)城是一个水上城市,一共有 \(N\) 个景点,有些景点之间会用一座桥连接.为了方便游 ...
- 洛谷 P1505 [国家集训队]旅游 树链剖分
目录 题面 题目链接 题目描述 输入输出格式 输入格式 输出格式 输入输出样例 输入样例: 输出样例: 说明 思路 AC代码 总结 题面 题目链接 P1505 [国家集训队]旅游 题目描述 Ray 乐 ...
- 2018.06.29 洛谷P1505 [国家集训队]旅游(树链剖分)
旅游 题目描述 Ray 乐忠于旅游,这次他来到了T 城.T 城是一个水上城市,一共有 N 个景点,有些景点之间会用一座桥连接.为了方便游客到达每个景点但又为了节约成本,T 城的任意两个景点之间有且只有 ...
- 洛谷P1505 [国家集训队]旅游
题目描述 \(Ray\) 乐忠于旅游,这次他来到了\(T\) 城.\(T\) 城是一个水上城市,一共有 \(N\) 个景点,有些景点之间会用一座桥连接.为了方便游客到达每个景点但又为了节约成本,\(T ...
- 洛谷P1505 [国家集训队]旅游(树剖+线段树)
传送门 这该死的码农题…… 把每一条边变为它连接的两个点中深度较浅的那一个,然后就是一堆单点修改/路径查询,不讲了 这里就讲一下怎么搞路径取反,只要打一个标记就好了,然后把区间和取反,最大最小值交换然 ...
- 模板—点分治A(容斥)(洛谷P2634 [国家集训队]聪聪可可)
洛谷P2634 [国家集训队]聪聪可可 静态点分治 一开始还以为要把分治树建出来……• 树的结构不发生改变,点权边权都不变,那么我们利用刚刚的思路,有两种具体的分治方法.• A:朴素做法,直接找重心, ...
- [洛谷P1527] [国家集训队]矩阵乘法
洛谷题目链接:[国家集训队]矩阵乘法 题目背景 原 <补丁VS错误>请前往P2761 题目描述 给你一个N*N的矩阵,不用算矩阵乘法,但是每次询问一个子矩形的第K小数. 输入输出格式 输入 ...
- 洛谷P1501 [国家集训队]Tree II(LCT,Splay)
洛谷题目传送门 关于LCT的其它问题可以参考一下我的LCT总结 一道LCT很好的练习放懒标记技巧的题目. 一开始看到又做加法又做乘法的时候我是有点mengbi的. 然后我想起了模板线段树2...... ...
- 洛谷P2619 [国家集训队2]Tree I(带权二分,Kruscal,归并排序)
洛谷题目传送门 给一个比较有逼格的名词--WQS二分/带权二分/DP凸优化(当然这题不是DP). 用来解决一种特定类型的问题: 有\(n\)个物品,选择每一个都会有相应的权值,需要求出强制选\(nee ...
随机推荐
- Duilib中各个类的简单介绍
DirectUI意为直接在父窗口上绘图(Paint on parent dc directly).即子窗口不以窗口句柄的形式创建(windowless),只是逻辑上的窗口,绘制在父窗口之上.微软的“D ...
- 牛客多校第六场 B Shorten IPv6 Address 模拟
题意: 给你一个二进制表示的IPv6地址,让你把它转换成8组4位的16进制,用冒号分组的表示法.单组的前导0可以省略,连续多组为0的可以用两个冒号替换,但是只允许替换一次.把这个地址通过这几种省略方式 ...
- http://www.jianshu.com/简书。
http://www.jianshu.com/ 简书,类似于博客园.也是一个交流平台.
- vue+element的el-menu组件实现路由跳转及当前项的设置
<el-menu router :default-active="$route.path" class="el-menu-vertical-demo" @ ...
- Spring NamedParameterJdbcTemplate详解(10)
NamedParameterJdbcTemplate和JdbcTemplate功能基本差不多.使用方法也类型.下面具体看下代码. db.properties 1 jdbc.user=root 2 jd ...
- JS时间比较大小
目录 1. 时间比较 2. 时间戳比较 3. 日期比较方法 4. 参考 1. 时间比较 var curTime = new Date(); //把字符串格式转化为日期类 var starttime = ...
- Amazon AWS EC2存储
- 那些使用VSCode写Python踩过的坑(Anaconda配置)
1. 如何在vscode上配置的配置方法请务必一定要直接参考官方文档Getting Started with Python in VS Code,不要去看什么杂七杂八的blog,要么过时要么不准确要么 ...
- Consul 集群搭建
搭建集群:.启动node1机器上的Consul (node1机器上执行) consul agent -data-dir /tmp/node1 -node=node1 -bind=192.168.0.1 ...
- iOS开发NSFetchedResultsController的使用CoreData和TableView数据同步更新
1.效果 2.代码 #import "ViewController.h" #import "Student+CoreDataProperties.h" #def ...