【题目背景】

于万人中万幸得以相逢,刹那间澈净明通。

成为我所向披靡的勇气和惶恐,裂山海,堕苍穹。

爱若执炬迎风,炽烈而哀恸,诸般滋味皆在其中。

韶华宛转吟诵,苍凉的光荣,急景凋年深情难共。

——银临《不老梦》

【问题描述】

扶苏翻遍了歌单却没有找到一首歌能做这个题的题目背景,于是放上了扶苏最喜欢 的一首《不老梦》。

与 Day1 的第二题一样,今天的第二题依然是一道树论题。(讨厌数论题没有之一!!)

我们定义一棵 n 个节点的树为一个有 n 个节点和 n-1 条边的无向连通图。

如果我们定义 u 是一颗树 T 的根,那么任意一个节点 v 到根的路径就是从 v 出发到 达点 u 的简单路径上所经过的点的点集。可以证明这样的简单路径有且仅有一条。

定义一个节点 x 是节点 y 的孩子,当且仅当 x 和 y 之间有边相连且 x 不在 y 到根的 路径中。如果 x 是 y 的孩子,那么定义 y 是 x 的家长节点。

如果我是 _rqy 那种毒瘤神仙的话,可能会问你每个节点的孩子数不超过 k 的 n 个节 点的带标号无根树一共有多少个,可惜这个问题我也不会,所以我不会问你这么毒瘤的 问题。

扶苏从一颗 n 个节点的树的 1 号节点出发,沿着树上的边行走。当然我们约定 1 号 节点是这棵树的根。他所行走的规定是:当扶苏在点 u 时,扶苏要么在 u 的孩子中选择 一个没有到达过得点 v 并行走到 v,要么选择回到 u的家长节点。

现在给每个节点一个权值 w,其中 i 号节点的权值为 wi。扶苏有一些石子,他想给 这棵树上的某一个节点放上石子。我们规定扶苏能在节点 u 放上石子当且仅当满足如下 条件:

1、扶苏当前在节点 u

2、对于 u 的所有孩子节点v,节点 v 被放上了 wv 颗石子。

但是,扶苏在任意时刻都可以取回任意节点的石子。

现在,扶苏想问问你对于每个节点,如果他想在 i 号节点上放 wi 颗石子,那么他一 开始需要准备多少石子。

【输入格式】

输入文件名为 yin.in。

输入文件中有且仅有一组数据,数据的第一行是一个整数 n 代表树的节点个数。

第二行有 n-1 个整数,第 i 个整数 pi 代表 i+1 号节点的家长节点的编号。

第三行有 n 个整数,第 i 个整数代表 wi。

【输出格式】

输出文件名为 yin.out。

输出一行 n 个整数,第 i 个整数代表想在 i 号节点上放 wi 颗石子需要准备的 石子个数。

显然给了测试点的数据,我们就可以逐个击破:

【测试点1】这个显然没有任何的技术含量啦,直接输出w1就可以啦。5pts get√

【测试点2-5】爆搜,搜出一个放石子的顺序,然后 O(n) 的 check 是否合法。时间复杂度 O(n!n)。期望得分 20 分。(但是我不会写嘤嘤嘤(╥╯^╰╥))

【测试点6-7】注意到根据题目规定的走法,在进入一个节点以后,必须遍历完它的整个子树, 否则一旦离开这个节点,再也无法进入这棵子树,从而导致该节点的某个孩子没能放 上石子,导致这个节点不能放上石子。同时又有每个节点放上石子以后,它的子树的 石子可以全部取回。设在节点 u 放石子需要有 ansu 个石子,则放完 u 以后可以取回 ansu-wu 个石子。

于是考虑影响问题答案的显然是从 u 进入每个孩子的顺序,由于最多有两个孩 子,直接比较一下就可以知道先进入哪个孩子更优秀了。时间复杂度 O(n),期望得分 10 分。

【测试点8-10】延续上一组测试点的思路,由于只有最多 5 个孩子,可以直接爆搜选孩子的顺 序,看看哪个更优秀。时间复杂度 O(n*x!),其中 x=5。期望得分 15 分。

【测试点11-14】(正解前导论,划重点)

【测试点15-20】可以发现上面的结论同样适用于树高更高的情况,于是在 dfs 回溯的时候对子节 点排序,即可算出该节点的答案,期望得分 30 分。

好了以下是充满了ZAY风起的STD(不用指针会死zay)

#include <cstdio>
#include <vector>
#include <algorithm> const int maxn = ; int n;
int MU[maxn], ans[maxn];
std::vector<int>son[maxn]; void dfs(const int u);
bool cmp(const int &_a, const int &_b); int main() {
freopen("yin.in", "r", stdin);
freopen("yin.out", "w", stdout);
scanf("%d", &n);
for (int i = , x; i <= n; ++i) {//直 接 i 从 2 开 始 枚 举,就 不 用 i+1 了
scanf("%d", &x);
son[x].push_back(i);//动 态 数 组 嘛 qwq
}
for (int i = ; i <= n; ++i) {
scanf("%d", MU + i);//指 针 可 海 星
}
dfs();
for (int i = ; i < n; ++i) {
printf("%d ", ans[i]);
}
printf("%d\n", ans[n]);
return ;
} void dfs(const int u) {
for (auto v : son[u]) {//遍历u结点的所有儿子,赋值给v
dfs(v);//先遍历所有儿子,最后再操作1
}
std::sort(son[u].begin(), son[u].end(), cmp);//将u结点的所有儿子按照ansi-wi不升序排序
int _ret = ;
//此时u结点所有的e子的ans都已经计算出来了
for (auto v : son[u]) {//对于叶子结点(没有e子),不进行这个for循环
//对于其它结点,
if (_ret >= ans[v]) {//显然由于遍历是有顺序的,当我们遍历到某些结点时,其之前遍历的
//结点的石子和(除去结点u的e子),(显然除了u结点所对应的儿子结点v们,这些v的儿子上的石子都可以删除啦)
//如果这些多出来的石子>某个未被遍历到的结点所需石子数,显然直接放过去就好了√
_ret -= ans[v];//这样剩余的石子就减少啦
//然后相当于ans[v]的贡献是0
} else {
ans[u] += ans[v] - _ret;//计算ans[u]:+它的e子结点所需要的石子-剩余石子
_ret = ans[v] - MU[v];//表示遍历完u的e子v这个结点对应滴子树后,可以收回除了u的e子v以外所有结点上的石头
//因此ret=ans[v]-MU[v]
}
}
ans[u] += std::max(, MU[u] - _ret);
//对于叶节点,ans[u]=0;
//对于其它结点
} inline bool cmp(const int &_a, const int &_b) {
return (ans[_a] - MU[_a]) > (ans[_b] - MU[_b]);//按照ansi-wi不升序排序
}

end-

【6.24校内test】T2 不老梦的更多相关文章

  1. 模拟赛DAY 2 T2不老梦

    [题目背景] 于万人中万幸得以相逢,刹那间澈净明通. 成为我所向披靡的勇气和惶恐,裂山海,堕苍穹. 爱若执炬迎风,炽烈而哀恸,诸般滋味皆在其中. 韶华宛转吟诵,苍凉的光荣,急景凋年深情难共. ——银临 ...

  2. 2019.6.24 校内测试 NOIP模拟 Day 2 分析+题解

    看到Day 2的题真的想打死zay了,忒难了QwQ~ T1 江城唱晚 这明显是个求方案数的计数问题,一般的套路是DP和组合数学. 正如题目中所说,这个题是一个 math 题.      ----zay ...

  3. 【7.24校内交流赛】T1&T2

    T1: 一个脑洞很大的题,将输入的所有数异或起来输出就好了: (话说我为什么这么喜欢用异或啊) #include<bits/stdc++.h> using namespace std; i ...

  4. [3.24校内训练赛by hzwer]

    来自FallDream的博客,未经允许,请勿转载,谢谢. ----------------------------------------------------------------------- ...

  5. 【6.24校内test】T1 江城唱晚

    [题目背景] 墙角那株海棠,是你种下的思念. 生死不能忘,高烛照容颜. 一曲江城唱晚,重忆当年坐灯前, 青衫中绣着你留下的线. ——银临<江城唱晚> [问题描述] 扶苏是个喜欢一边听古风歌 ...

  6. 18清明校内测试T2

    一道数论好题(math) Time Limit:1000ms   Memory Limit:128MB 题目描述 rsy最近在研究欧几里得算法,不会的同学可以去看下课件以及代码…… 现在她想到了一个新 ...

  7. 2019.7.9 校内测试 T2 极值问题

    这一次是交流测试?边交流边测试(滑稽 极值问题 乍一看这是一道数学题,因为1e9的数据让我暴力的心退却. 数学又不好,不会化简式子嘞,咋办? 不怕,咱会打表找规律.(考场上真的是打表找出了规律,打表打 ...

  8. 2019.6.28 校内测试 T2 【音乐会】二重变革

    看到这个题之后,一个很暴力很直接的想法就是贴上题目中的代码然后交上去走人,但是很显然这是会TLE+MLE的,想想谁会这么傻把主要代码给你QwQ~: 其实这段代码是想告诉你一件事:用序列中的大数减去小数 ...

  9. 【6.24校内test】T3 棠梨煎雪

    [题目背景] 岁岁花藻檐下共将棠梨煎雪. 自总角至你我某日辗转天边. 天淡天青,宿雨沾襟. 一年一会信笺却只见寥寥数言. ——银临<棠梨煎雪> [问题描述] 扶苏正在听<棠梨煎雪&g ...

随机推荐

  1. 2018 焦作网络赛 L Poor God Water ( AC自动机构造矩阵、BM求线性递推、手动构造矩阵、矩阵快速幂 )

    题目链接 题意 : 实际上可以转化一下题意 要求求出用三个不同元素的字符集例如 { 'A' .'B' .'C' } 构造出长度为 n 且不包含 AAA.BBB CCC.ACB BCA.CAC CBC ...

  2. Mybatis 结果映射下划线转驼峰

    mybatis 结果映射下划线转驼峰 Spring Boot 配置: #下划线转驼峰 mybatis.configuration.map-underscore-to-camel-case=true m ...

  3. HDU 6071 Lazy Running (最短路)

    题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=6071 题解 又是一道虐信心的智商题... 首先有一个辅助问题,这道题转化了一波之后就会化成这个问题: ...

  4. Acwing:102. 最佳牛围栏(前缀和 + 二分)

    农夫约翰的农场由 NN 块田地组成,每块地里都有一定数量的牛,其数量不会少于1头,也不会超过2000头. 约翰希望用围栏将一部分连续的田地围起来,并使得围起来的区域内每块地包含的牛的数量的平均值达到最 ...

  5. Exchanger 原理

    Exchanger(交换者)是一个用于线程间协作的工具类.Exchanger用于进行线程间的数据交换.它提供一个同步点,在这个同步点两个线程可以交换彼此的数据.这两个线程通过exchange方法交换数 ...

  6. JAVA源文件中可以包含多个类

    可以的,一个“.java”源文件里面可以包含多个类,但是只允许有一个public类,并且类名必须和文件名一直:例如: package test; public class test { } class ...

  7. 20道HTML基础面试题(附答案)

    以下是我整理的一些HTML的基础面试体,并自己整理了答案. 1 DOCTYPE有什么作用?标准模式与混杂模式如何区分?它们有何意义? 告诉浏览器使用哪个版本的HTML规范来渲染文档.DOCTYPE不存 ...

  8. 第六周学习总结&第四次实验报告

    第六周学习总结&第四次实验报告 学习总结 这周我们简单的学习了一点点关于接口的内容,接口是Java中最重要的概念之一,接口可以理解为一个特殊的类, 里面由全局常量和公共的抽象方法组成,接口摆脱 ...

  9. Java JDBC 基础

    JDBC API 包含以下几个核心部分: 1:JDBC 驱动 2:Connections (连接) 3:Statements (声明) 4:Result Sets (结果集) JDBC: 打开数据库连 ...

  10. Python2.x与Python3.x的主要区别(转)

    python2.x和python3.x版本有很大的差异,除了依赖包的名称变化很大外,其主要差异总结如下: 1)print函数 Python3中,print函数的括号是必须的,Python2是可选的. ...