题目链接:

传送门

题目分析:

树剖板,支持单点修改,区间取反,区间求最大值/最小值/和

区间取反取两次等于没取,维护一个\(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 [国家集训队]旅游的更多相关文章

  1. 洛谷 P1505 [国家集训队]旅游 解题报告

    P1505 [国家集训队]旅游 题目描述 \(\tt{Ray}\) 乐忠于旅游,这次他来到了\(T\)城.\(T\)城是一个水上城市,一共有 \(N\) 个景点,有些景点之间会用一座桥连接.为了方便游 ...

  2. 洛谷 P1505 [国家集训队]旅游 树链剖分

    目录 题面 题目链接 题目描述 输入输出格式 输入格式 输出格式 输入输出样例 输入样例: 输出样例: 说明 思路 AC代码 总结 题面 题目链接 P1505 [国家集训队]旅游 题目描述 Ray 乐 ...

  3. 2018.06.29 洛谷P1505 [国家集训队]旅游(树链剖分)

    旅游 题目描述 Ray 乐忠于旅游,这次他来到了T 城.T 城是一个水上城市,一共有 N 个景点,有些景点之间会用一座桥连接.为了方便游客到达每个景点但又为了节约成本,T 城的任意两个景点之间有且只有 ...

  4. 洛谷P1505 [国家集训队]旅游

    题目描述 \(Ray\) 乐忠于旅游,这次他来到了\(T\) 城.\(T\) 城是一个水上城市,一共有 \(N\) 个景点,有些景点之间会用一座桥连接.为了方便游客到达每个景点但又为了节约成本,\(T ...

  5. 洛谷P1505 [国家集训队]旅游(树剖+线段树)

    传送门 这该死的码农题…… 把每一条边变为它连接的两个点中深度较浅的那一个,然后就是一堆单点修改/路径查询,不讲了 这里就讲一下怎么搞路径取反,只要打一个标记就好了,然后把区间和取反,最大最小值交换然 ...

  6. 模板—点分治A(容斥)(洛谷P2634 [国家集训队]聪聪可可)

    洛谷P2634 [国家集训队]聪聪可可 静态点分治 一开始还以为要把分治树建出来……• 树的结构不发生改变,点权边权都不变,那么我们利用刚刚的思路,有两种具体的分治方法.• A:朴素做法,直接找重心, ...

  7. [洛谷P1527] [国家集训队]矩阵乘法

    洛谷题目链接:[国家集训队]矩阵乘法 题目背景 原 <补丁VS错误>请前往P2761 题目描述 给你一个N*N的矩阵,不用算矩阵乘法,但是每次询问一个子矩形的第K小数. 输入输出格式 输入 ...

  8. 洛谷P1501 [国家集训队]Tree II(LCT,Splay)

    洛谷题目传送门 关于LCT的其它问题可以参考一下我的LCT总结 一道LCT很好的练习放懒标记技巧的题目. 一开始看到又做加法又做乘法的时候我是有点mengbi的. 然后我想起了模板线段树2...... ...

  9. 洛谷P2619 [国家集训队2]Tree I(带权二分,Kruscal,归并排序)

    洛谷题目传送门 给一个比较有逼格的名词--WQS二分/带权二分/DP凸优化(当然这题不是DP). 用来解决一种特定类型的问题: 有\(n\)个物品,选择每一个都会有相应的权值,需要求出强制选\(nee ...

随机推荐

  1. C++在#include命令中,用〈 〉和“”有什么区别

    使用尖括号表示在包含文件目录中去查找(包含目录是由用户在设置环境时设置的),而不在源文件目录去查找: 使用双引号则表示首先在当前的源文件目录中查找,若未找到才到包含目录中去查找.  

  2. CF1086F Forest Fires

    CF1086F Forest Fires 有点意思的题目 直接统计每个格子的val是非常难办的.很难知道每秒新出来多少个格子 设$F[i]$表示,前i时刻覆盖的格子的数量 则,$ans=\sum_{i ...

  3. vuex的简单介绍

    .vuex的定义 )Vuex 是一个专门为 Vue.js 应用程序开发的状态管理模式,使用插件的形式引进项目中 )集中存储和管理应用的所有组件的状态,并以相应的规则保证状态以一种可预测的方式发生变化 ...

  4. idea从github中pull或者push成功之后ssm项目全部controller报红色下划线的解决方案

    某次从github上pull下来之后,报出如下一堆红色波浪线错误 解决方法: 在随便一个红色波浪线处,按下alt+enter,点击add maven dependency, 选中所有,点击添加即可,此 ...

  5. windows安装vscode,配置golang环境

    出现的问题: 进行如下命令进行目录切换:cd %GOPATH%\src\github.com\golang我这里的GOPATH是在D:\GoPath,大家这里一定要注意些如果src目录下面没有gith ...

  6. PAT甲级——A1125 Chain the Ropes【25】

    Given some segments of rope, you are supposed to chain them into one rope. Each time you may only fo ...

  7. 20.multi_case06

    # coding:utf-8 import asyncio # 通过create_task()方法 async def a(t): print('-->', t) await asyncio.s ...

  8. Django杂篇(1)

    目录 Django杂篇(1) bulk_create Pagination 创建多对多表关系的常用方法 form校验组件的应用 渲染页面 展示错误信息 校验数据 常用字段 Django杂篇(1) 这里 ...

  9. (转)Nginx+Php-fpm运行原理详解

    一.代理与反向代理 现实生活中的例子 1.正向代理:访问google.com 如上图,因为google被墙,我们需要vpn翻墙才能访问google.com. vpn对于“我们”来说,是可以感知到的(我 ...

  10. php7 mysql_pconnect() 缺失 解决方法

    php7 兼容 MySQL 相关函数 PHP7 废除了 ”mysql.dll” ,推荐使用 mysqli 或者 pdo_mysql http://PHP.net/manual/zh/mysqlinfo ...