[洛谷]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 ...
随机推荐
- VS2010-MFC(常用控件:静态文本框)
转自:http://www.jizhuomi.com/software/179.html 关于对话框的使用和各种通用对话框的介绍就到此为止,从本节开始将讲解各种常用控件的用法.常用控件主要包括:静态文 ...
- 初识OpenCV-Python - 006: 图像的几何变换
本次小节学习了图像的变换,主要应用到如下方法: cv2.resize(), cv2.warpAffine(), cv2.getRotationMatrix2D(), cv2.getAffineTran ...
- 【Oracle】如何在查询视图时使用索引
通常我们使用hint来固定查询计划选择走表的索引 固定表的连接等等,但是如果第一层查询的是视图呢? yang@rac1>CREATE TABLE TA (ID NUMBER, NAME VARC ...
- 16-1-es5
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- go string和[ ]byte
https://www.cnblogs.com/zhangboyu/p/7623712.html
- 模拟实现call、apply
1. 知识点补充: 首先在模拟实现前,先Mark一些我之前不知道的知识: a. eval(string)函数:可计算某个字符串,并执行其中的JavaScript代码 其中,string是必需传入的待计 ...
- Luogu P4158 [SCOI2009]粉刷匠(dp+背包)
P4158 [SCOI2009]粉刷匠 题意 题目描述 \(windy\)有\(N\)条木板需要被粉刷.每条木板被分为\(M\)个格子. 每个格子要被刷成红色或蓝色. \(windy\)每次粉刷,只能 ...
- Entity Framework Code First 模式-建立多对多联系
Entity Framework 在建立多对多的联系时,会生成一个中间表,用来表示这个多对多的关系.这和数据库设计时从概念模型到逻辑模型转化时,多对多的关系不能和任何一端的实体合并,需要将关系也转化为 ...
- QVaraint类
一.QVaraint简介 QVaraint类作为一个包含了大多数Qt普通类型的的联合体类.因为c++联合体没有构造函数和析构函数的,许多继承的Qt类不能够在联合体当中使用.(联合体当中的变量共 ...
- 180608发现的一个有趣的Douyin-Bot项目
今日发现的github 项目 Douyin-Bot 抖音机器人发现漂亮小姐姐 :sweat_smile: 在开发者模式下,找到指针位置,开启之后,当点击屏幕时,可以获取点击屏幕处的xy坐标了 我使用的 ...