[洛谷]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 ...
随机推荐
- (转)Android中px与dip,sp与dip等的转换工具类
功能 通常在代码中设置组件或文字大小只能用px,通过这个工具类我们可以把dip(dp)或sp为单位的值转换为以px为单位的值而保证大小不变.方法中的参数请参考http://www.cnblogs.co ...
- Http学习(二)
使用首部字段是为了给浏览器和服务器提供报文主体大小.所使用语言.认证信息等 4种首部字段类型 通用首部字段 请求首部字段 响应首部字段 实体首部字段 详细说明: HTTP首部字段类型 通用首部字段: ...
- Hibernate之OID
在关系数据库中,主键用来识别记录,并保证每天记录的唯一性.在Java语言中,通过比较两个变量所引用对象的内存地址是否相同,或者比较两变量引用的对象是否相等.Hibernate为了解决两者之间的不同,使 ...
- HUD1686-Oulipo-kmp模板题/哈希模板题
The French author Georges Perec (1936–1982) once wrote a book, La disparition, without the letter 'e ...
- day 55 Django基础五之django模型层(一)单表操作
Django基础五之django模型层(一)单表操作 本节目录 一 ORM简介 二 单表操作 三 章节作业 四 xxx 一 ORM简介 MVC或者MVC框架中包括一个重要的部分,就是ORM,它 ...
- 《Python之BMI计算》
<Python之BMI计算> 前段时间写了个 BMI 因为刚刚开始学 有几个错误 第一个: 厘米我当时也没注意因为觉得去掉0.00的话后面1866666666是正确的BMI值 刚刚去看看去 ...
- selenium基础(鼠标和键盘事件)
selenium鼠标和键盘的操作事件 webdriver常见的几种操作方法 clear():清楚文本文字 send_keys(values):模拟按键输入,values是输入的内容 click():单 ...
- java笔试之字符逆序(二)
与字符逆序(一)不同,句子逆序,单词顺序.例如:I am a student 输出为student a am I. 想法:先字符串句子逆序,然后针对每个单词再逆序一遍. package test; i ...
- UMP系统架构 LVS
- opencv编译:opencv 3.4.1 编译 contrib模块,增加人脸识别
start cmake-gui select the opencv source code folder and the folder where binaries will be built (th ...