P4253 SCOI2015 小凸玩密室
P4253 SCOI2015 小凸玩密室
一道紫色的 dp。
思路
首先读题:
要保证任意时刻所有被点亮的灯泡必须连通
在点亮一个灯泡后必须先点亮其子树所有灯泡才能点亮其他灯泡
考虑设 \(g[u][1]\) 为 \(u\) 子树第一个被选中的是子树的根的代价,\(g[u][0]\) 为 \(u\) 子树内第一个选中的点不是子树的根(全局第一个选中的点在 \(u\) 子树内且不为 \(u\))。
我们发现转移与距离有关,遂加一维。
设 \(g[u][1][j]\) 为 \(u\) 子树第一个被选中的是子树的根的代价,且 \(u\) 子树选完后,最后一个被选的点到 \(u\) 的距离为 \(j\)。
\(g[u][0][j]\) 为 \(u\) 子树内第一个选中的点不是子树的根,且 \(u\) 子树选完后,最后一个被选的点到 \(u\) 的距离为 \(j\)。
反复思考题目中的两个条件,你会发现,每个子树内只有最后一个选中的是叶子节点才是合法情况。
但再思考一下,发现如果一个节点只有一个儿子,还是有可能自己成为子树内最后一个选中的节点。
不过这是一棵完全二叉树。
那么出现这种情况的点最多只会有一个,这种点特殊考虑就可以了。
我们可以写出一般情况下的二叉树转移方程:
设 \(v_1\)、\(v_2\) 为 \(u\) 的两个儿子,两条边的长度为 \(d[v_1]\) 和 \(d[v_2]\),\(dis[u]\) 为 \(u\) 子树中叶子节点到 \(u\) 节点的距离集合。
\]
其中 \(i\in dis[v_2]\)。
这条方程的含义是:点亮 \(u\) 以后先去点亮 \(v_1\),再从 \(v_1\) 子树内的叶子节点点亮 \(v_2\),最后在 \(v_2\) 的叶子节点点完 \(u\) 子树,此时距离 \(v_2\) 节点是 \(i\),那么距离 \(u\) 节点就是 \(i+d[v_2]\) 了。
至于为什么都是 \(g[v][1][j]\) 在转移呢,因为父亲都子树内第一个点亮的了,那么想要点亮其它点肯定要先点亮儿子呀。
类似的可以写出:
\]
其中 \(i\in dis[v_1]\)。
然后我们考虑针对 \(g[u][0][i]\) 的转移,此时自己的儿子可以是他所在子树内先点亮的,也可以是后点亮的,都是合法的,对于另外一个儿子肯定就是先点亮根的喽,那么有转移:
\]
类似的我们需要把 \(v_1\) 和 \(v_2\) 换过来,在写一遍这个转移,下面就不再写了。
观察转移方程,注意到我们每个方程中含 \(j\) 的那一项,对于每一个 \(u\) 来说都是一个固定值。
所以每一次提前求出含 \(j\) 项的最小值,然后枚举 \(i\) 就可以转移了。
考虑时间复杂度:
每次转移都相当于遍历一遍自己子树内叶子节点到自己的距离。
所以每一层的转移复杂度都是 \(O(叶子节点个数)\)。
然而根据完全二叉树,层数不会超过 \(\log n\) 层,总复杂度就是 \(O(叶子节点个数 \log n)\)。
CODE
#include<bits/stdc++.h>
using namespace std;
#define ll long long
const int maxn=2e5+5;
int n;
ll d[maxn],a[maxn];
vector<ll>vec[maxn],dis[maxn];
unordered_map<ll,ll>g[maxn][2];
inline void solve1(int u,int v1,int v2)//1 的转移
{
ll res=1e18;
for(auto j:dis[v1])//提前求含 j 项
res=min(res,g[v1][1][j]+a[v2]*(j+d[v2]+d[v1])+a[v1]*d[v1]);
for(auto j:dis[v2])
{
if(g[u][1][j+d[v2]]==0) g[u][1][j+d[v2]]=1e18;
g[u][1][j+d[v2]]=min(res+g[v2][1][j],g[u][1][j+d[v2]]);
}
}
inline void solve2(int u,int v1,int v2)//0 的转移
{
ll res0=1e18,res1=1e18;
for(auto j:dis[v1])
res0=min(res0,g[v1][0][j]+(j+d[v1])*a[u]+d[v2]*a[v2]),res1=min(res1,g[v1][1][j]+(j+d[v1])*a[u]+d[v2]*a[v2]);
for(auto j:dis[v2])
{
if(g[u][0][j+d[v2]]==0) g[u][0][j+d[v2]]=1e18;
g[u][0][j+d[v2]]=min(res0+g[v2][1][j],g[u][0][j+d[v2]]);
g[u][0][j+d[v2]]=min(res1+g[v2][1][j],g[u][0][j+d[v2]]);
}
}
inline void dfs(int u)
{
int v1=u*2,v2=u*2+1;
if(v1>n&&v2>n){vec[u].push_back(u);dis[u].push_back(0);return ;}
dfs(v1);
if(v2<=n) dfs(v2);
else
{
g[u][1][d[v1]]=d[v1]*a[v1];g[u][0][0]=d[v1]*a[u];
g[u][1][0]=1e18;g[u][0][d[v1]]=1e18;
vec[u].push_back(v1);vec[u].push_back(v2);dis[u].push_back(d[v1]);dis[u].push_back(0);
return ;
}
solve1(u,v1,v2);solve1(u,v2,v1);//v1,v2 转移方程是一样的
solve2(u,v1,v2);solve2(u,v2,v1);
vec[u].insert(vec[u].end(),vec[v1].begin(),vec[v1].end());
vec[u].insert(vec[u].end(),vec[v2].begin(),vec[v2].end());
dis[u].resize(vec[u].size());//处理 u 与叶子节点距离
for(int i=0;i<vec[v1].size();i++) dis[u][i]=dis[v1][i]+d[v1];
for(int i=0;i<vec[v2].size();i++) dis[u][i+vec[v1].size()]=dis[v2][i]+d[v2];
}
int main()
{
scanf("%d",&n);
for(int i=1;i<=n;i++) scanf("%lld",&a[i]);
for(int i=2;i<=n;i++) scanf("%lld",&d[i]);
dfs(1);
ll ans=1e18;
for(auto j:dis[1]) ans=min(ans,min(g[1][1][j],g[1][0][j]));
printf("%lld\n",ans);
}
后记
竟然是第一道独立做出来的紫色 dp。
P4253 SCOI2015 小凸玩密室的更多相关文章
- LUOGU P4253 [SCOI2015]小凸玩密室(树形dp)
传送门 解题思路 玄学树形\(dp\),题目描述极其混乱...看错了两次题,设首先根据每次必须点完子树里的灯才能点别的,那么点灯情况只有两种,第一种是点到某一个祖先,第二种是点到某一个祖先的兄弟.所以 ...
- [BZOJ4446]SCoi2015 小凸玩密室 树形DP(烧脑高能预警)
4446: [Scoi2015]小凸玩密室 Time Limit: 10 Sec Memory Limit: 128 MB Description 小凸和小方相约玩密室逃脱,这个密室是一棵有n个节点 ...
- bzoj 4446: [Scoi2015]小凸玩密室
Description 小凸和小方相约玩密室逃脱,这个密室是一棵有n个节点的完全二叉树,每个节点有一个灯泡.点亮所有灯 泡即可逃出密室.每个灯泡有个权值Ai,每条边也有个权值bi.点亮第1个灯泡不需要 ...
- BZOJ4446:[SCOI2015]小凸玩密室(树形DP)
Description 小凸和小方相约玩密室逃脱,这个密室是一棵有n个节点的完全二叉树,每个节点有一个灯泡.点亮所有灯泡即可逃出密室. 每个灯泡有个权值Ai,每条边也有个权值bi.点亮第1个灯泡不需要 ...
- BZOJ4446 [Scoi2015]小凸玩密室 【树形Dp】
题目 小凸和小方相约玩密室逃脱,这个密室是一棵有n个节点的完全二叉树,每个节点有一个灯泡.点亮所有灯 泡即可逃出密室.每个灯泡有个权值Ai,每条边也有个权值bi.点亮第1个灯泡不需要花费,之后每点亮4 ...
- [bzoj4446] [loj#2009] [Scoi2015] 小凸玩密室
Description 小凸和小方相约玩密室逃脱,这个密室是一棵有 \(n\) 个节点的完全二叉树,每个节点有一个灯泡.点亮所有灯泡即可逃出密室.每个灯泡有个权值 \(Ai\) ,每条边也有个权值 \ ...
- BZOJ4446: [Scoi2015]小凸玩密室
用ui,j表示走完i的子树后走到i的深度为j的祖先的兄弟的最小代价: 用vi,j表示走完i的子树后走到i的深度为j的祖先的最小代价,用u算出v. 枚举起点,计算答案. #include<bits ...
- 2019.03.26 bzoj4446: [Scoi2015]小凸玩密室(树形dp)
传送门 题意简述: 给一棵完全二叉树,有点权aia_iai和边权,每个点有一盏灯,现在要按一定要求点亮: 任意时刻点亮的灯泡必须连通 点亮一个灯泡后必须先点亮其子树 费用计算如下:点第一盏灯不要花费 ...
- BZOJ.4446.[SCOI2015]小凸玩密室(树形DP)
BZOJ LOJ 洛谷 (下面点亮一个灯泡就说成染色了,感觉染色比较顺口... 注意完全二叉树\(\neq\)满二叉树,点亮第一个灯泡\(\neq\)第一次点亮一号灯泡,根节点应该就是\(1\)... ...
- BZOJ4446 SCOI2015小凸玩密室(树形dp)
设f[i][j]为由根进入遍历完i子树,最后一个到达的点是j时的最小代价,g[i][j]为由子树内任意一点开始遍历完i子树,最后一个到达的点是j时的最小代价,因为是一棵完全二叉树,状态数量是nlogn ...
随机推荐
- Ubuntu 连接联想 M7206W 打印机
联想很多打印机其实是 Brother 打印机贴牌(OEM): 所以有时候 Brother 的打印机驱动也可以用于联想打印机. 将打印机连接到 Wi-Fi: 在打印机上,按下 功能 按钮进入设置,通过上 ...
- 记 某List.sort()后排序结果异常
背景:某次查看日志,发现数据不符合预期,希望获取的是降序排序,但是部分数据是乱序的 已知List.sort()方法应该不会出异常,所以应该是判断先后方法出问题了 果然,因为一开始写代码时,没有考虑到差 ...
- Android 执行shell命令 非root
一个开源的项目已经完成了这个功能 app_process-shell-use,这个项目网上讲解的,很多,我这边就把操作时一些简要步骤说明一下 但是该方式有缺点:必须要先USB连接,debug的时候,运 ...
- LCD屏幕显示PNG图像
正点原子LCD屏幕显示PNG图像 本文概要 这段时间在学习正点原子的IMX6ULL开发板,在应用编程中有一个代码练习是需要在LCD屏幕上显示PNG图像,但由于我的屏幕参数和教程中的有些出入,于是经过自 ...
- WebRTC 简单入门与实践
一.前言 WebRTC 技术已经广泛在各个行业及场景中被应用,但对多数开发者来说,实时音视频及相关技术却是比较不常接触到的. 做为一名 Web 开发者,WebRTC 这块的概念着实花了不少时间才搞明白 ...
- Stooges – AI 孙燕姿
很久没有写文章了... 最近听了 AI 孙燕姿 的一首歌 <一场游戏一场梦>, 有感而发. 所以特别写一篇记入一下我的这半年很火的 AI 的想法.
- GPUStack正式发布: 为大模型而生的开源GPU集群管理器
经过 Seal 研发团队几个月来持续的开发和测试,我们满怀期待及热情地发布新的产品 GPUStack,GPUStack 是一个用于运行 LLM(大型语言模型)的开源 GPU 集群管理器.尽管如今大语言 ...
- 【Simpleperf】Android的CPU分析,性能优化利器
很多时候,写代码是一件很爽的事情,但最后需要对APP进行瘦身.性能分析却是一件很棘手的事情.当需要对APP的性能进行分析时,Simpleperf是一个简单快捷的选择. 正文开始前,先奉上官方的资料: ...
- JOI 2018 Final
A - ストーブ (Stove) 有 \(n\) 个客人将要来访,第 \(i\) 个客人的来访时间为 \([a_i, a_i + 1]\),保证 \(\forall i \in [1, n), a_i ...
- C#/.NET/.NET Core技术前沿周刊 | 第 9 期(2024年10.07-10.13)
前言 C#/.NET/.NET Core技术前沿周刊,你的每周技术指南针!记录.追踪C#/.NET/.NET Core领域.生态的每周最新.最实用.最有价值的技术文章.社区动态.优质项目和学习资源等. ...