题意

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. sql新语句

    SQL语句查找重复数据的写法:  select partner_id,* from sale_origin where partner_id in (select   partner_id from  ...

  2. Linux shell ftp命令下载文件 根据文件日期

    需求:ftp获取远程数据的文件,根据文件的创建时间点下载文件. 可以自行扩展根据文件的大小等其他需求. 知识点总结: 1.获取文件的时间: ls -lrt|awk '{print $6" & ...

  3. C# out关键词应用

    C#的out关键词,即是方法内赋值. 返回处理后的结果.打个比喻,有一个宽度的需要按比例缩放.标准宽度为88,如宽度大于这个标准宽度的话,按照0.8进行缩放.如果小于标准宽度,输出的结果没变化. 此时 ...

  4. excel的宏与VBA入门——代码调试

    直接介绍重点: 常用的操作是导航栏的逐句与断点: 添加断点:调试->切换断点 单步运行:调试->逐句 查看变量的窗口:视图->本地窗口

  5. 51nod 抽卡大赛

    抽卡大赛 链接 分析: $O(n^4)$的做法比较好想,枚举第i个人选第j个,然后背包一下,求出有k个比他大的概率. 优化: 第i个人,选择一张卡片,第j个人选的卡片大于第i个人的概率是$p_j$,那 ...

  6. C#编程:从控制台读取数字的两种方式

    有时需要从控制台输入数字,就用到前面介绍的内容,数据转换,如:int num=int.Pares(Console.ReadLine()); int num=Convert.ToInt32(Consol ...

  7. 五年.net程序员Java学习之路

    大学毕业后笔者进入一家外企,做企业CRM系统开发,那时候开发效率最高的高级程序语言,毫无疑问是C#.恰逢公司也在扩张,招聘了不少.net程序员,笔者作为应届生,也乐呵呵的加入到.net程序员行列中. ...

  8. Docker容器学习梳理 - 容器硬盘热扩容

    前面已介绍了docker很多知识点的操作记录,今天这里梳理下docker容器空间扩展的操作.默认情况下,物理机下创建的docker容器的空间是10G(虚拟机下创建的docker容器空间就是虚拟机的空间 ...

  9. python基础学习笔记(十二)

    模块 前面有简单介绍如何使用import从外部模块获取函数并且为自己的程序所用: >>> import math >>> math.sin(0) #sin为正弦函数 ...

  10. 关于dreamweaver的软件测评

    最近在用javascript编写程序,于是便用到了dreamweaver .所以,想写一个关于dreamweaver的软件测评. 学生本人使用的是dreamweaver 8.首先,谈谈本人使用感受,打 ...