题解

真真正正是个码农题,不过很套路,熟练就打得很快,不过要用点维护边的信息在 \(\text{LCA}\) 出要注意,不能处理此点的信息

\(Code\)

#include<cstdio>
#include<iostream>
#include<cstring>
using namespace std; const int N = 2e5 + 5;
int n, m, h[N]; struct edge{int to, nxt, w, id;}e[N << 1];
inline void add(int u, int v, int w, int id)
{
static int tot = 0;
e[++tot] = edge{v, h[u], w, id}, h[u] = tot;
} int top[N], fa[N], dfn[N], dep[N], siz[N], son[N], ver[N], edg[N];
void dfs1(int x)
{
siz[x] = 1;
for(register int i = h[x]; i; i = e[i].nxt)
{
int v = e[i].to;
if (v == fa[x]) continue;
edg[v] = e[i].w, ver[e[i].id] = v, fa[v] = x, dep[v] = dep[x] + 1, dfs1(v), siz[x] += siz[v];
if (siz[v] > siz[son[x]]) son[x] = v;
}
}
void dfs2(int x)
{
static int dfc = 0;
dfn[x] = ++dfc;
if (son[x]) top[son[x]] = top[x], dfs2(son[x]);
for(register int i = h[x]; i; i = e[i].nxt)
{
int v = e[i].to;
if (v == fa[x] || v == son[x]) continue;
top[v] = v, dfs2(v);
}
} struct Tree{
#define ls (p << 1)
#define rs (ls | 1)
const int INF = 0x3f3f3f3f;
int sum[N << 2], mn[N << 2], mx[N << 2], tag[N << 2]; inline Tree(){memset(mn, 0x3f3f3f3f, sizeof mn), memset(mx, -0x3f3f3f3f, sizeof mx);}
void change(int p){sum[p] *= -1, mx[p] *= -1, mn[p] *= -1, swap(mx[p], mn[p]), tag[p] ^= 1;}
void pushup(int p){sum[p] = sum[ls] + sum[rs], mn[p] = min(mn[ls], mn[rs]), mx[p] = max(mx[ls], mx[rs]);}
void pushdown(int p)
{
if (!tag[p]) return;
change(ls), change(rs), tag[p] ^= 1;
} void update_node(int p, int l, int r, int x, int v)
{
if (l == r)
{
sum[p] = mn[p] = mx[p] = v;
if (l == 1) mn[p] = INF, mx[p] = -INF;
return;
}
pushdown(p);
int mid = (l + r) >> 1;
if (x <= mid) update_node(ls, l, mid, x, v);
else update_node(rs, mid + 1, r, x, v);
pushup(p);
} void update_rev(int p, int l, int r, int x, int y)
{
if (x <= l && r <= y) return void(change(p));
pushdown(p);
int mid = (l + r) >> 1;
if (x <= mid) update_rev(ls, l, mid, x, y);
if (y > mid) update_rev(rs, mid + 1, r, x, y);
pushup(p);
}
inline void tree_rev(int x, int y)
{
int fx = top[x], fy = top[y];
while (fx ^ fy)
{
if (dep[fx] > dep[fy]) update_rev(1, 1, n, dfn[fx], dfn[x]), x = fa[fx], fx = top[x];
else update_rev(1, 1, n, dfn[fy], dfn[y]), y = fa[fy], fy = top[y];
}
if (dep[x] > dep[y]) swap(x, y);
if (x == y) return;
update_rev(1, 1, n, dfn[x] + 1, dfn[y]);
} int query_sum(int p, int l, int r, int x, int y)
{
if (x <= l && r <= y) return sum[p];
pushdown(p);
int mid = (l + r) >> 1, ret = 0;
if (x <= mid) ret += query_sum(ls, l, mid, x, y);
if (y > mid) ret += query_sum(rs, mid + 1, r, x, y);
return ret;
}
inline int tree_sum(int x, int y)
{
int fx = top[x], fy = top[y], ret = 0;
while (fx ^ fy)
{
if (dep[fx] > dep[fy]) ret += query_sum(1, 1, n, dfn[fx], dfn[x]), x = fa[fx], fx = top[x];
else ret += query_sum(1, 1, n, dfn[fy], dfn[y]), y = fa[fy], fy = top[y];
}
if (dep[x] > dep[y]) swap(x, y);
if (x == y) return ret;
return ret + query_sum(1, 1, n, dfn[x] + 1, dfn[y]);
} int query_max(int p, int l, int r, int x, int y)
{
if (x <= l && r <= y) return mx[p];
pushdown(p);
int mid = (l + r) >> 1, ret = -INF;
if (x <= mid) ret = max(ret, query_max(ls, l, mid, x, y));
if (y > mid) ret = max(ret, query_max(rs, mid + 1, r, x, y));
return ret;
}
inline int tree_max(int x, int y)
{
int fx = top[x], fy = top[y], ret = -INF;
while (fx ^ fy)
{
if (dep[fx] > dep[fy]) ret = max(ret, query_max(1, 1, n, dfn[fx], dfn[x])), x = fa[fx], fx = top[x];
else ret = max(ret, query_max(1, 1, n, dfn[fy], dfn[y])), y = fa[fy], fy = top[y];
}
if (dep[x] > dep[y]) swap(x, y);
if (x == y) return ret;
return max(ret, query_max(1, 1, n, dfn[x] + 1, dfn[y]));
} int query_min(int p, int l, int r, int x, int y)
{
if (x <= l && r <= y) return mn[p];
pushdown(p);
int mid = (l + r) >> 1, ret = INF;
if (x <= mid) ret = min(ret, query_min(ls, l, mid, x, y));
if (y > mid) ret = min(ret, query_min(rs, mid + 1, r, x, y));
return ret;
}
inline int tree_min(int x, int y)
{
int fx = top[x], fy = top[y], ret = INF;
while (fx ^ fy)
{
if (dep[fx] > dep[fy]) ret = min(ret, query_min(1, 1, n, dfn[fx], dfn[x])), x = fa[fx], fx = top[x];
else ret = min(ret, query_min(1, 1, n, dfn[fy], dfn[y])), y = fa[fy], fy = top[y];
}
if (dep[x] > dep[y]) swap(x, y);
if (x == y) return ret;
return min(ret, query_min(1, 1, n, dfn[x] + 1, dfn[y]));
}
}seg; int main()
{
scanf("%d", &n);
for(register int i = 1, u, v, w; i < n; i++)
scanf("%d%d%d", &u, &v, &w), ++u, ++v, add(u, v, w, i), add(v, u, w, i);
dfs1(1), top[1] = 1, dfs2(1);
for(register int i = 1; i <= n; i++) seg.update_node(1, 1, n, dfn[i], edg[i]);
scanf("%d", &m);
char op[5];
for(int u, v; m; --m)
{
scanf("%s%d%d", op, &u, &v), ++u, ++v;
if (op[0] == 'C') seg.update_node(1, 1, n, dfn[ver[u - 1]], v - 1);
else if (op[0] == 'N') seg.tree_rev(u, v);
else if (op[0] == 'S') printf("%d\n", seg.tree_sum(u, v));
else if (op[1] == 'A') printf("%d\n", seg.tree_max(u, v));
else printf("%d\n", seg.tree_min(u, v));
}
}

Luogu P1505.[国家集训队]旅游的更多相关文章

  1. LUOGU P1505 [国家集训队]旅游 (树链剖分+线段树)

    传送门 解题思路 快被调死的码农题,,,其实就是一个边权下放到点权的线段树+树剖. #include<iostream> #include<cstdio> #include&l ...

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

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

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

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

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

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

  5. P1505 [国家集训队]旅游

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

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

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

  7. P1505 [国家集训队]旅游[树剖]

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

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

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

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

    题目链接: 传送门 题目分析: 树剖板,支持单点修改,区间取反,区间求最大值/最小值/和 区间取反取两次等于没取,维护一个\(rev\ tag\),每次打标记用\(xor\)打,记录是否需要翻转,\( ...

  10. luoguP1505 [国家集训队]旅游(真的毒瘤)

    luogu P1505 [国家集训队]旅游 题目 #include<iostream> #include<cstdio> #include<cstdlib> #in ...

随机推荐

  1. MyBatis详解(一)

    MyBatis简单介绍 [1]MyBatis是一个持久层的ORM框架[Object Relational Mapping,对象关系映射],使用简单,学习成本较低.可以执行自己手写的SQL语句,比较灵活 ...

  2. java 常用的jar包下载地址

    Eclipse: http://www.eclipse.org/downloads/packages/all Spring: http://Framework: http://repo.spring. ...

  3. Dojo dijit/Tree的使用以及样式设置

    如果什么都不设置,默认使用dojo自带的Tree,样式模板使用claro的,效果是这样的. 1.无论是不是叶子节点,前面总要带个+号,必须要点击下才消失. 2.点击树或者某个节点是,回出现蓝色边框. ...

  4. Spring Boot整合log4j实战(一):排除自带依赖、日志重定向、测试类验证

    〇.参考资料 1.springboot整合log4j全过程详解 https://blog.csdn.net/m0_60845963/article/details/123307232 2.Spring ...

  5. STM32标准库中GPIO_ReadInputData与GPIO_ReadInputDataBit的区别

    GPIO_ReadInputData读的是GPIOx的整个IDR寄存器的数据,返回一个十六位数,对应IDR寄存器的十六位.反映GPIOx所有端口的电平状态,所以参数只用传入GPIOx. uint16_ ...

  6. ArcObjects SDK开发 021 开发框架搭建-FrameWork包设计

    1.框架引擎部分 引擎模块其实就是之前我们说的App-Command-Tool模块,通过这个模块,把系统的主干框架搭建起来. 其中大部分出现在菜单以及工具条上的按钮都会继承这个框架定义ICommand ...

  7. 网络监测工具之Zabbix的搭建与测试方法(一)

    简介 Zabbix是一个基于WEB界面的提供分布式系统监视以及网络监视功能的企业级的开源解决方案,它能监视各种网络参数,保证服务器系统的安全运营:并提供灵活的通知机制以让系统管理员快速定位/解决存在的 ...

  8. 【博学谷学习记录】超强总结,用心分享|前端CSS总结(一)

    CSS总结(一) shift+alt,选中多行 外链式 <link rel="stylesheet" href="./my.css"> 1 选择器 ...

  9. [OpenCV实战]18 Opencv中的单应性矩阵Homography

    目录 1 介绍 1.1 什么是Homography 1.2 使用Homography进行图像对齐 1.3 Homography的应用-全景拼接 2 Homography的计算 3 总结 4 参考 &l ...

  10. JAVA中使用最广泛的本地缓存?Ehcache的自信从何而来 —— 感受来自Ehcache的强大实力

    大家好,又见面了. 本文是笔者作为掘金技术社区签约作者的身份输出的缓存专栏系列内容,将会通过系列专题,讲清楚缓存的方方面面.如果感兴趣,欢迎关注以获取后续更新. 作为<深入理解缓存原理与实战设计 ...