题意

click here

题解

我们首先考虑答案是个什么样的东西, 不难 发现每个点可以单独计算它的贡献。

令每个点 \(i\) 崛起次数为 \(a_i\) 。

假设一个点子树的 \(\sum a_i\) 分别为 \(b_1,b_2,\dots,b_k\) ,令 \(S = a_i + \sum b_j\) 。

那么这个点的答案为

\[\min (2(S - \max(\max\{b_j\}, a_i)), S - 1)
\]

至于为什么是这样可以简单说明下:

\(S - 1\) :显然是这个点的答案的上界,除了第一次,后面每一次最多对这个点贡献一次。

\(2(S - \max(\max\{b_j\}, a_i))\) :不难发现,我们总可以找到一种方案使得 \(S - \max\) 那种与 剩下的 \(\max(\max\{b_j\}, a_i)\) 交错出现,使得这个答案取得上界,然后每次会存在两种贡献。

这样我们就可以得到一个 \(O(n)\) 的 \(dp\) 了。


我们考虑如何动态维护这个 \(dp\) 。

不难发现这个操作及其类似于 Link_Cut_Tree 中的 Access 操作。

我们令 \(b_u = \sum_{v \in child(u)} a_v\) 也就是 \(u\) 的子树 \(a\) 和。

我们考虑维护这个东西,不难发现每次给一个点的 \(a_u\) 加上 \(v\) ,相当于把这个点到根的 \(b_u\) 加上 \(v\) 。

然后考虑如何维护一个点的贡献,如果 \(u\) 存在一个儿子 \(v\) 使得 \(b_v \times 2 > b_u + 1\) 那么我们定义 \(v \to u\) 为实边。

其余的边都为虚边。不难发现这些实边会对于 \(u\) 存在 \(2(b_u - b_v)\) 的贡献。( \(a\) 不可能存在贡献,因为 \(b_v\) 已经占据一半了)

然后虚边的贡献就是 \(\min(b_u - 1, 2(b_u - a_u))\) 。

这个可以自己列列不等式,讨论讨论就行了。

然后我们每次 Access 操作就是将链上的一些点加权,并且更换虚实边就行了,重新计算贡献就行了。

这个直接用支持加法标记的 LCT 维护就行了。

不难发现一个点到根的实边最多是 \(\log w\) 条,因为每条实边会使得权值至少翻倍。

所以最后复杂度就是 \(O(n + q\log w)\) 的。

总结

有些题,我们先找出它的一些巧妙性质以及结论,然后考虑用数据结构维护。(这似乎也是出题的好思路?)

代码

不太会写的话还是建议看看代码的。。

#include <bits/stdc++.h>

#define For(i, l, r) for(register int i = (l), i##end = (int)(r); i <= i##end; ++i)
#define Fordown(i, r, l) for(register int i = (r), i##end = (int)(l); i >= i##end; --i)
#define Set(a, v) memset(a, v, sizeof(a))
#define Cpy(a, b) memcpy(a, b, sizeof(a))
#define debug(x) cout << #x << ": " << x << endl
#define DEBUG(...) fprintf(stderr, __VA_ARGS__) using namespace std; typedef long long ll;
inline bool chkmin(ll &a, ll b) {return b < a ? a = b, 1 : 0;}
inline bool chkmax(ll &a, ll b) {return b > a ? a = b, 1 : 0;} inline int read() {
int x = 0, fh = 1; char ch = getchar();
for (; !isdigit(ch); ch = getchar()) if (ch == '-') fh = -1;
for (; isdigit(ch); ch = getchar()) x = (x << 1) + (x << 3) + (ch ^ 48);
return x * fh;
} void File() {
#ifdef zjp_shadow
freopen ("374.in", "r", stdin);
freopen ("374.out", "w", stdout);
#endif
} const int N = 4e5 + 1e3; int n, m;
vector<int> G[N]; int son[N]; ll Ans, ans[N], b[N], a[N];
inline void ReCalc(int u) {
Ans -= ans[u];
ans[u] = son[u] ? 2 * (b[u] - b[son[u]]) : b[u] - 1;
if (a[u] * 2 > b[u] + 1) ans[u] = 2 * (b[u] - a[u]);
Ans += ans[u];
} #define ls(o) ch[o][0]
#define rs(o) ch[o][1] namespace Link_Cut_Tree { int ch[N][2], fa[N]; inline bool get(int o) { return o == rs(fa[o]); } inline bool is_root(int o) { return o != ls(fa[o]) && o != rs(fa[o]); } inline void Rotate(int v) {
int u = fa[v], t = fa[u], d = get(v);
fa[ch[u][d] = ch[v][d ^ 1]] = u;
fa[v] = t; if (!is_root(u)) ch[t][rs(t) == u] = v;
fa[ch[v][d ^ 1] = u] = v;
} ll tag[N];
inline void Add(int o, ll uv) { if (o) b[o] += uv, tag[o] += uv; } inline void Push_Down(int o) {
if (!tag[o]) return ;
Add(ls(o), tag[o]);
Add(rs(o), tag[o]); tag[o] = 0;
} inline void Push_All(int o) {
if (!is_root(o)) Push_All(fa[o]); Push_Down(o);
} inline void Splay(int o) {
Push_All(o);
for (; !is_root(o); Rotate(o))
if (!is_root(fa[o])) Rotate(get(o) != get(fa[o]) ? o : fa[o]);
} inline int Get_Root(int o) {
while (ls(o)) Push_Down(o), o = ls(o); return o;
} inline void Access(int o, int uv) {
for (int t = 0; o; o = fa[t = o]) {
Splay(o);
b[o] += uv; Add(ls(o), uv); if (son[o]) {
Push_All(son[o]);
if (b[son[o]] * 2 <= b[o] + 1) son[o] = rs(o) = 0;
}
int to = Get_Root(t);
if (b[to] * 2 > b[o] + 1) son[o] = to, rs(o) = t;
ReCalc(o);
}
} } void Dfs_Init(int u, int fa = 0) {
Link_Cut_Tree :: fa[u] = fa; b[u] = a[u]; int to = 0;
for (int v : G[u]) if (v != fa) {
Dfs_Init(v, u); b[u] += b[v];
if (b[v] > b[to]) to = v;
}
if (b[to] * 2 > b[u]) son[u] = Link_Cut_Tree :: rs(u) = to; ReCalc(u);
} int main () { File(); n = read(); m = read();
For (i, 1, n) a[i] = read();
For (i, 1, n - 1) {
int u = read(), v = read();
G[u].push_back(v); G[v].push_back(u);
}
Ans = 0; Dfs_Init(1); printf ("%lld\n", Ans);
For (i, 1, m) {
int pos = read(), val = read();
a[pos] += val; Link_Cut_Tree :: Access(pos, val);
printf ("%lld\n", Ans);
} return 0;
}

LOJ #2434. 「ZJOI2018」历史(LCT)的更多相关文章

  1. @loj - 2434@ 「ZJOI2018」历史

    目录 @description@ @solution@ @accepted code@ @details@ @description@ 九条可怜是一个热爱阅读的女孩子. 这段时间,她看了一本非常有趣的 ...

  2. LOJ2434. 「ZJOI2018」历史 [LCT]

    LOJ 思路 第一眼看似乎没有什么思路,试着套个DP上去:设\(dp_x\)表示只考虑\(x\)子树,能得到的最大答案. 合并的时候发现只有\(x\)这个点有可能做出新的贡献,而做出新贡献的时候必然是 ...

  3. 「ZJOI2018」历史(LCT)

    「ZJOI2018」历史(LCT) \(ZJOI\) 也就数据结构可做了-- 题意:给定每个点 \(access\) 次数,使轻重链切换次数最大,带修改. \(30pts:\) 挺好想的.发现切换次数 ...

  4. 「ZJOI2018」历史

    「ZJOI2018」历史 前置知识 \(\text{LCT}\) 维护子树信息,考虑辅助树上一个节点的子树信息只是其代表的这一段链的信息,设 \(S(u)\) 为节点 \(u\) 的子树信息,那么在辅 ...

  5. Loj #2529. 「ZJOI2018」胖

    Loj #2529. 「ZJOI2018」胖 题目描述 Cedyks 是九条可怜的好朋友(可能这场比赛公开以后就不是了),也是这题的主人公. Cedyks 是一个富有的男孩子.他住在著名的 The P ...

  6. 题解 「ZJOI2018」历史

    题目传送门 Description 九条可怜是一个热爱阅读的女孩子. 这段时间,她看了一本非常有趣的小说,这本小说的架空世界引起了她的兴趣. 这个世界有 \(n\) 个城市,这 \(n\) 个城市被恰 ...

  7. LOJ#2230. 「BJOI2014」大融合

    LOJ#2230. 「BJOI2014」大融合 题目描述 小强要在$N$个孤立的星球上建立起一套通信系统.这套通信系统就是连接$N$个点的一个树.这个树的边是一条一条添加上去的. 在某个时刻,一条边的 ...

  8. Loj #2192. 「SHOI2014」概率充电器

    Loj #2192. 「SHOI2014」概率充电器 题目描述 著名的电子产品品牌 SHOI 刚刚发布了引领世界潮流的下一代电子产品--概率充电器: 「采用全新纳米级加工技术,实现元件与导线能否通电完 ...

  9. Loj #3096. 「SNOI2019」数论

    Loj #3096. 「SNOI2019」数论 题目描述 给出正整数 \(P, Q, T\),大小为 \(n\) 的整数集 \(A\) 和大小为 \(m\) 的整数集 \(B\),请你求出: \[ \ ...

随机推荐

  1. 3.5《想成为黑客,不知道这些命令行可不行》(Learn Enough Command Line to Be Dangerous)—第三章小结

    本章使用的重要命令总结在Table 5中 命令 描述 示例 curl 与URL交互 $ curl -O example.com which 指出程序的在计算机的路径 $ echo bar >&g ...

  2. DIV实现水平或垂直滚动条

    添加样式: 在html中,需要创建2层div来实现.一个div包含另一个div: 效果:

  3. vue 动态加载组建

    <component :is="comp1"></component> data () { return { comp1:'', } } require.e ...

  4. [Spark][Hive]外部文件导入到Hive的例子

    外部文件导入到Hive的例子: [training@localhost ~]$ cd ~[training@localhost ~]$ pwd/home/training[training@local ...

  5. Elasticsearch 系列文章汇总(持续更新...)

    系列文章列表 Query DSL Query DSL 概要,MatchAllQuery,全文查询简述 Match Query Match Phrase Query 和 Match Phrase Pre ...

  6. .NET持续集成与自动化部署之路第一篇——半天搭建你的Jenkins持续集成与自动化部署系统

    .NET持续集成与自动化部署之路第一篇(半天搭建你的Jenkins持续集成与自动化部署系统) 前言     相信每一位程序员都经历过深夜加班上线的痛苦!而作为一个加班上线如家常便饭的码农,更是深感其痛 ...

  7. bootstrap datetimepicker 格式化yyyymmdd时,无法读取yyyymmdd格式

    不知为何,java程序员爱用yyyymmdd格式化日期?导致bootstrap datetimepicker无法解析正确的日期 发现js中yyyymmdd不是正常能够解析的日期 查看datetimep ...

  8. 深入理解USB流量数据包的抓取与分析

    0x01 问题提出 在一次演练中,我们通过wireshark抓取了一个如下的数据包,我们如何对其进行分析? 0x02 问题分析 流量包是如何捕获的? 首先我们从上面的数据包分析可以知道,这是个USB的 ...

  9. grep精确匹配搜索某个单词的用法 (附: grep高效用法小结))

    grep(global search regular expression(RE) and print out the line,全面搜索正则表达式并把行打印出来)是一种强大的文本搜索工具,它能使用正 ...

  10. 对我们最常用的软件QQ的看法

    QQ聊天软件是我使用的第一款聊天软件,早在我上小学6年级的时候就开始接触这款软件了,可以说是陪伴我最久的一款软件. 相对于其他的聊天软件,QQ更加的方便,使用简单,界面也好操作,所以我爱上了这款软件. ...