题解:P9704 「TFOI R1」Tree Home

题解:P9704 「TFOI R1」Tree Home

Leo2011 警示后人

本文会带你保姆级推公式,应该是题解中推公式最详细的一篇才不是因为我数学和 OI 都太菜了,因而用到了大量数学公式,可能会加载较慢,请耐心等待。

挺逆天的一道数学题。

这个 函数有 个参数明显不好处理,那我们拿出做初一化简求值题的感觉直接化简:

这个部分很简单,乘法分配律然后去括号抵消一些项就好了。

接着把 代入:

式子很长,但思路很简单,就是先用完全平方公式()和完全立方公式()以及乘法分配律代入替换拆括号(第一、二步)然后疯狂合并同类项发现基本都抵消了(第三步),然后就剩下了这么个可可爱爱简单的式子(暴力出奇迹.jpg)。

ps:有没有大佬有简单的方法化简呢?手敲 KaTeX 化简了十几分钟 Qwq。

说句闲话:原式中出现了大量的 ,化简完竟然和它无关…… 化简这个式子估计占了题目难度的一半。

于是题目要求的式子就是 。然后鉴于有绝对值,继续利用初一知识分类讨论,有以下四种情况:

然后我们发现每个式子都有两个项下标一样,可以一起处理。于是我们令 ,然后把上面四个式子整理一下,分别是:

如果需要最大化原式那么我们需要让被减数尽量大,减数尽量小。由于区间被指定了所以这说到底是个 RMQ,而由于没有修改因此可以通过 个 ST 表维护。

终于我们有了最终思路:

  1. 预处理出 数组
  2. 利用 ST 表查询
  3. 比较得出结果

时间复杂度分为两部分,预处理 数组可以用一个 dfs,时间复杂度 ,由于是一棵树所以 ,最终忽略常数复杂度就是 。ST 表预处理为 ,查询为 ,可以通过。


ACCode with 注释
line-numbers
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
/*Code by Leo2011*/
#include <bits/stdc++.h>

#define INF 0x3f3f3f3f
#define EPS 1e-8
#define FOR(i, l, r) for (ll(i) = (l); (i) <= (r); ++(i))
#define log printf
#define IOS \
ios::sync_with_stdio(false); \
cin.tie(nullptr); \
cout.tie(nullptr);

using namespace std;

typedef __int128 i128;
typedef long long ll;
typedef pair<ll, ll> PII;

const ll N = 2e5 + 10;
ll n, t, u, v, w, a[N], l1, l2, r1, r2, lg2[N], dis[N], st1[30][N], st2[30][N], st3[30][N], st4[30][N];
vector<PII> g[N];

template <typename T>

inline T read() {
T sum = 0, fl = 1;
char ch = getchar();
for (; !isdigit(ch); ch = getchar())
if (ch == '-') fl = -1;
for (; isdigit(ch); ch = getchar()) sum = sum * 10 + ch - '0';
return sum * fl;
}

template <typename T>

inline void write(T x) {
if (x < 0) {
putchar('-'), write<T>(-x);
return;
}
static T sta[35];
ll top = 0;
do { sta[top++] = x % 10, x /= 10; } while (x);
while (top) putchar(sta[--top] + 48);
}

// 4 个 ST 表,很长但是基本复制粘贴
void init() {
lg2[2] = 1;
FOR(i, 1, n) {
st1[0][i] = st2[0][i] = dis[i] * dis[i] * dis[i] + a[i];
st3[0][i] = st4[0][i] = dis[i] * dis[i] * dis[i] - a[i];
}
FOR(i, 3, n) lg2[i] = lg2[i >> 1] + 1;
FOR(i, 1, lg2[n]) {
ll len = 1 << i;
FOR(j, 1, n - len + 1) {
st1[i][j] = max(st1[i - 1][j], st1[i - 1][j + (len >> 1)]);
st2[i][j] = min(st2[i - 1][j], st2[i - 1][j + (len >> 1)]);
st3[i][j] = max(st3[i - 1][j], st3[i - 1][j + (len >> 1)]);
st4[i][j] = min(st4[i - 1][j], st4[i - 1][j + (len >> 1)]);
}
}
}

// 好的函数命名可以方便写代码和调试
ll queryMax1(ll l, ll r) {
ll k = lg2[r - l + 1], len = 1 << k;
return max(st1[k][l], st1[k][r - len + 1]);
}

ll queryMin1(ll l, ll r) {
ll k = lg2[r - l + 1], len = 1 << k;
return min(st2[k][l], st2[k][r - len + 1]);
}

ll queryMax2(ll l, ll r) {
ll k = lg2[r - l + 1], len = 1 << k;
return max(st3[k][l], st3[k][r - len + 1]);
}

ll queryMin2(ll l, ll r) {
ll k = lg2[r - l + 1], len = 1 << k;
return min(st4[k][l], st4[k][r - len + 1]);
}

// 要先预处理出 d 数组
void dfs(ll now, ll fa) {
for (auto v : g[now]) {
if (v.first == fa) continue;
dis[v.first] = dis[now] + v.second; // 要继承父亲的 d 然后加上自己的
dfs(v.first, now);
}
}

int main() {
n = read<ll>(), t = read<ll>();
FOR(i, 1, n) a[i] = read<ll>();
FOR(i, 2, n) {
u = read<ll>(), v = read<ll>(), w = read<ll>();
g[u].push_back({v, w}), g[v].push_back({u, w});
}
dfs(1, -1);
init(); // ST 表和线段树都要初始化!
while (t--) {
l1 = read<ll>(), r1 = read<ll>(), l2 = read<ll>(), r2 = read<ll>();
write<ll>(max({queryMax1(l1, r1) - queryMin1(l2, r2), queryMax1(l2, r2) - queryMin1(l1, r1), queryMax2(l1, r1) - queryMin2(l2, r2), queryMax2(l2, r2) - queryMin2(l1, r1)})), putchar('\n'); // 输出的 4 种情况取 max
}
return 0;
}

AC 记录~

代码很长,但是非常模块化,写起来不是很慢。

理解万岁!

  • 标题: 题解:P9704 「TFOI R1」Tree Home
  • 作者: Leo2011
  • 创建于 : 2025-08-07 10:12:32
  • 更新于 : 2025-08-26 21:21:38
  • 链接: https://www.leo2011.eu.org/2025/08/07/ti-jie-p9704-tfoi-r1-tree-home/
  • 版权声明: 本文章采用 CC BY-NC-SA 4.0 进行许可。
评论
Nickname
Email
Website
  • OωO
  • |´・ω・) ノ
  • ヾ (≧∇≦*) ゝ
  • (☆ω☆)
  • (╯‵□′)╯︵┴─┴
  •  ̄﹃ ̄
  • (/ω\)
  • ∠( ᐛ 」∠)_
  • (๑•̀ㅁ•́ฅ)
  • →_→
  • ୧(๑•̀⌄•́๑)૭
  • ٩(ˊᗜˋ*)و
  • (ノ °ο°) ノ
  • (´இ皿இ`)
  • ⌇●﹏●⌇
  • (ฅ´ω`ฅ)
  • (╯°A°)╯︵○○○
  • φ( ̄∇ ̄o)
  • ヾ (´・ ・`。) ノ "
  • (ง ᵒ̌皿ᵒ̌)ง⁼³₌₃
  • (ó﹏ò。)
  • Σ(っ °Д °;) っ
  • (,,´・ω・)ノ"(´ っ ω・`。)
  • ╮(╯▽╰)╭
  • o(*////▽////*)q
  • >﹏<
  • ( ๑´•ω•) "(ㆆᴗㆆ)
  • 😂
  • 😀
  • 😅
  • 😊
  • 🙂
  • 🙃
  • 😌
  • 😍
  • 😘
  • 😜
  • 😝
  • 😏
  • 😒
  • 🙄
  • 😳
  • 😡
  • 😔
  • 😫
  • 😱
  • 😭
  • 💩
  • 👻
  • 🙌
  • 🖕
  • 👍
  • 👫
  • 👬
  • 👭
  • 🌚
  • 🌝
  • 🙈
  • 💊
  • 😶
  • 🙏
  • 🍦
  • 🍉
  • 😣
  • 颜文字
  • Emoji
  • Bilibili
0 comments
No comment