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 ...
随机推荐
- Kubernetes-10:Ingress-nginx介绍及演示
Ingress介绍 Ingress是什么? ingress 是除了 hostport nodeport clusterIP以及云环境专有的负载均衡器外的访问方式,官方提供了Nginx ingres ...
- IDEA maven 项目 如何获取项目离线运行所需的全部依赖( .m2格式)
背景:maven项目要将整个项目的依赖移植到某无法联网服务器进行测试,需要项目离线运行所需的全部依赖 步骤: 1. 首先需要有项目源码,解压后,使用IDEA Open Project 2. 在Sett ...
- C# 模拟http请求出现 由于系统缓冲区空间不足或队列已满,不能执行套接字上的操作[windows服务器]
系统里面用到C#模拟Http请求,上线到服务器后,发现日志中大量出现"由于系统缓冲区空间不足或队列已满,不能执行套接字上的操作" 或"通常每个套接字地址(协议/网络地址/ ...
- Graph 学习
Graph basic terms 里面介绍了常见的一些基本概念,如 directed/undirected, weighted, cyclic/acyclic, Adjacency Matrix, ...
- JDBC,SQL注入,事务,C3P0于Druid连接池(最详细解析)
JDBC JDBC(Java DataBase Connectivty,Java数据库连接)API,是一种用于执行Sql语句的Java API,可以为关系型数据库提供统一的访问,其由一组Java编写的 ...
- C# – class, filed, property, const, readonly, get, set, init, required 使用基础
前言 心血来潮,这篇讲点基础的东西. Field 比起 Property,Field 很不起眼,你若问 JavaScript,它甚至都没有 Field. 但在 C#,class 里头真正装 value ...
- springboot多文件上传、删除、下载到项目本地
package com.example.demo.document; import cn.hutool.core.io.FileUtil; import cn.hutool.core.io.IoUti ...
- 【赵渝强老师】Oracle RAC集群的概念
一.什么是Oracle RAC(Real Application Cluster)? Oracle RAC 是一个具有共享缓存架构的集群数据库,它克服了传统的无共享方法和共享磁盘方法的限制,为您的所有 ...
- USB硬件特性(速度、名称、供电)
USB传输速度 USB1.0版本,USB LS(Low Speed低速),速度1.5Mbps. USB1.1版本,USB FS(Full Speed全速),速度12Mbps. USB2.0版本,USB ...
- 深度学习卷积、全连接层、深度可分离层参数量和FLOPs计算公式
来源 卷积: 输入尺寸 ,卷积核的大小为 * ,输出的尺寸大小为 参数量 (1)不考虑bias:(2)考虑bias: FLOPs (1)不考虑bias 解释:先计算输出的中一个元素需要的计算量, ...