题意 n个点组成一棵树, 带有点权。 求最长不降的路径的长度, 且路径上最大值最小值之差不超过D。

显然是树分治, 但是分治之后如何维护答案呢。

假设当前重心为g, 分别记录g出发不降路径的长度,以及最大值, 和不升路径的长度以及最小值。

这里用到一个map和二分, 线段树也可以, 但是如果用线段树还要考虑负值, 再加上线段树的clear以及稍微暴力的查询。  常数大小不好说。

 #include <bits/stdc++.h>
using namespace std;
typedef pair <int, int> pii;
const int maxn = 1e5 + ;
vector <int> G[maxn];
int val[maxn], siz[maxn], n, D;
bool centroid[maxn];
void init() {
for (int i = ; i < maxn; i++) {
G[i].clear();
}
memset(centroid, false, sizeof centroid);
}
void dfs(int u, int father){
siz[u] = ;
for (int v: G[u]){
if (v != father && !centroid[v]){
dfs(v, u);
siz[u] += siz[v];
}
}
}
pii FindCentroid(int u, int father, int t) {
pii res = make_pair(INT_MAX, -);
int s = , m = ;
for (int v: G[u]) {
if (v == father || centroid[v]) {
continue;
}
res = min(res, FindCentroid(v, u, t));
m = max(m, siz[v]);
s += siz[v];
}
siz[u] = s;
m = max(m, t-s);
return min(res, make_pair(m, u));
}
map <int, int> work;
void update(int v, int len) {
auto it = work.lower_bound(v);
if (it != work.end() && it->second >= len) {
return;
}
work[v] = len;
}
void dfs_up(int u, int father, int d) {
if (val[u] > val[father]) {
return;
}
update(val[u], d);
for (int v: G[u]) {
if (v != father && !centroid[v]) {
dfs_up(v, u, d+);
}
}
}
int res;
void dfs_down(int u, int father, int d) {
if (val[u] < val[father]) {
return;
}
auto it = work.lower_bound(val[u]-D);
if (it != work.end()) {
res = max(res, it->second++d);
}
for (int v: G[u]) {
if (!centroid[v] && v != father) {
dfs_down(v, u, d+);
}
}
} void preSolve(int g, vector <int> &son) {
work.clear();
work[val[g]] = ;
for (int v: son) {
if (val[v] >= val[g]) {
dfs_down(v, g, );
}
if (val[v] <= val[g]) {
dfs_up(v, g, );
}
}
}
void solve(int u) {
dfs(u, );
int g = FindCentroid(u, , siz[u]).second;
vector <int> son;
for (int v: G[g]) {
if (!centroid[v]) {
son.push_back(v);
}
}
preSolve(g, son);
reverse(son.begin(), son.end());
preSolve(g, son);
centroid[g] = true;
for (int v: G[g]) {
if (!centroid[v]) {
solve(v);
}
}
}
int main() {
// freopen("in.txt", "r", stdin);
int T, cas = ;
scanf ("%d", &T);
while (T--) {
init();
scanf ("%d%d", &n, &D);
for (int i = ; i <= n; i++) {
scanf ("%d", val+i);
}
for (int i = ; i < n; i++) {
int u, v;
scanf ("%d%d", &u, &v);
G[u].push_back(v);
G[v].push_back(u);
}
res = ;
solve();
printf("Case #%d: %d\n", cas++, res);
}
return ;
}

UVALive 7148 LRIP 14年上海区域赛K题 树分治的更多相关文章

  1. UVAlive7141 BombX 14年上海区域赛D题 线段树+离散化

    题意:一个无限大的棋盘, 有n个小兵, 给出了n个兵的坐标, 现在有一个长为width 高为height的炸弹放在棋盘上, 炸弹只能上下左右平移, 不能旋转. 且放炸弹的区域不能含有士兵, 炸弹可以一 ...

  2. hdu5080:几何+polya计数(鞍山区域赛K题)

    /* 鞍山区域赛的K题..当时比赛都没来得及看(反正看了也不会) 学了polya定理之后就赶紧跑来补这个题.. 由于几何比较烂写了又丑又长的代码,还debug了很久.. 比较感动的是竟然1Y了.. * ...

  3. ACM 2015年上海区域赛A题 HDU 5572An Easy Physics Problem

    题意: 光滑平面,一个刚性小球,一个固定的刚性圆柱体 ,给定圆柱体圆心坐标,半径 ,小球起点坐标,起始运动方向(向量) ,终点坐标 ,问能否到达终点,小球运动中如果碰到圆柱体会反射. 学到了向量模板, ...

  4. hdu 4463 有一条边必须加上 (2012杭州区域赛K题)

    耐克店 和 苹果店必须相连 Sample Input42 30 01 00 -1 1 -10 Sample Output3.41 # include <iostream> # includ ...

  5. ACM学习历程——ZOJ 3829 Known Notation (2014牡丹江区域赛K题)(策略,栈)

    Description Do you know reverse Polish notation (RPN)? It is a known notation in the area of mathema ...

  6. Peekaboo(2019年上海网络赛K题+圆上整点)

    目录 题目链接 题意 思路 代码 题目链接 传送门 题意 你的位置在\(O(0,0)\),\(A\)的位置为\((x_1,y_1)\),\(B\)的位置为\((x_2,y_2)\),现在已知\(a=O ...

  7. 2018南京区域赛K题 Kangaroo Puzzle——随机&&乱搞

    题意 在 n * m 的平面上有若干个袋鼠和墙(1为袋鼠,0为墙),每次可以把所有袋鼠整体往一个方向移动一步(不能走出边界和不能走到墙),为在不超过50000步的情况下能否把全部袋鼠聚集在同一个位置. ...

  8. 2019 ICPC 上海区域赛总结

    2019上海区域赛现场赛总结 补题情况(以下通过率为牛客提交): 题号 标题 已通过代码 通过率 我的状态 A Mr. Panda and Dominoes 点击查看 5/29 未通过 B Prefi ...

  9. HDU-5532//2015ACM/ICPC亚洲区长春站-重现赛-F - Almost Sorted Array/,哈哈,水一把区域赛的题~~

    F - Almost Sorted Array Time Limit:2000MS     Memory Limit:262144KB     64bit IO Format:%I64d & ...

随机推荐

  1. vs中debug和release版本的区别(转)

    vs中的程序有debug和release两个版本,Debug通常称为调试版本,通过一系列编译选项的配合,编译的结果通常包含调试信息,而且不做任何优化,以为开发 人员提供强大的应用程序调试能力.而Rel ...

  2. kill session真的能杀掉进程吗

    session1 确认sidSYS @ prod > select userenv('sid') from dual; USERENV('SID')-------------- 144 sess ...

  3. MySQL数据库迁移详细步骤(转)

    ========================================================================================== 一.背景简介 == ...

  4. 开发自己的cordova插件

    如果还没有配置过cordova环境,首先要下载nodejs,(下载地址https://nodejs.org/)下载完毕安装. 控制台: 1.输入npm -v 确定是否装上了 2.输入sudo npm ...

  5. 读懂IL代码(二)

    上一篇提到了最基本的IL代码,应该是比较通俗易懂的,所以有了上一篇的基础之后,这篇便要深入一点点的来讲述了. 首先我必须再来说一些重要的概念: Evaluation Stack(评估栈):这是由.NE ...

  6. 向RichTextBox控件不停的AppendText数据时,如何把光标的焦点始终显示到最后

    上面是csdn上的一个网友的问题,我的一个实现如下://让文本框获取焦点this.richTextBoxInfo.Focus();//设置光标的位置到文本尾this.richTextBoxInfo.S ...

  7. asp.net 网站和asp.net Web 应用程序的一处不同

    环境为:VS2008Team+.net3.5 asp.net 网站前台页面<%= %>这样绑定可以,asp.net Web 应用程序就不可以 示例代码如下: 1.asp.net网站 < ...

  8. poj 1273.PIG (最大流)

    网络流 关键是建图,思路在代码里 /* 最大流SAP 邻接表 思路:基本源于FF方法,给每个顶点设定层次标号,和允许弧. 优化: 1.当前弧优化(重要). 1.每找到以条增广路回退到断点(常数优化). ...

  9. 原生js实现回到顶部

    网页可见区域宽: document.body.clientWidth;网页可见区域高: document.body.clientHeight;网页可见区域宽: document.body.offset ...

  10. 火狐无法访问本机IIS部署的网站,弹出:此地址使用了一个通常用于网络浏览以外目的的端口.出于安全原因,Firefox 取消了该请求 的解决办法

    关于火狐浏览器访问本机IIS部署的网站弹出“此地址使用了一个通常用于网络浏览以外目的的端口.出于安全原因,Firefox 取消了该请求”这个错误(错误截图如下): 解决方法如下: 1.打开火狐浏览器, ...