[BZOJ4446]SCoi2015 小凸玩密室 树形DP(烧脑高能预警)
4446: [Scoi2015]小凸玩密室
Time Limit: 10 Sec Memory Limit: 128 MB
Description
Input
Output
Sample Input
5 1 2
2 1
Sample Output
HINT
对于100%的数据,1≤N≤2×105,1<Ai,Bi≤10^5
(其实我是回来补暑假没写完的题解的)
题解:
(首先我经过激烈的思想斗争,认为那个点亮灯泡的花费计算只与上一个有关(其实是因为前4个太难考虑了...)
事实证明这样做是对的..就是只考虑上一个:(
有了之前做非线性DP的经验,我一开始想的还是合并类型
但是发现数据范围不太对....这似乎是一个介于O(n)和O(n2)范围内的DP
我们考虑怎么定义状态,以及怎么转移.
一开始我想的是f[i][j]表示"走完以i为根的子树之后走去j点的最小花费"
但是我发现这个MLE了,2e5开不下
但是这又是一颗完全二叉树,所以我们考虑能不能应用他的一些性质
我们观察到,点灯泡的起点没有确定,但是题目有这样的两个限制:
要保证任意时刻所有被点亮的灯泡必须连通.
在点亮一个灯泡后必须先点亮其子树所有灯泡才能点亮其他灯泡
这样的话,某一个点被点亮的时候只能有三种转移的情况:从儿子走来,从兄弟走来和从父亲走来(废话)
因此,当某一个子树被完全点亮之后,我们就要去某一个他的祖先,或者是他某个祖先的儿子
我们发现上面这两个都与这个节点的祖先有关(这里我们把自己也看成自己的祖先)
由于这是一棵二叉树,我们完全可以通过位运算(左移,右移和异或)来计算出某个节点的某个深度的祖先.
这样,第二维完全不用是O(n)的:我们可以把第二维设为深度,即走完以i为根的子树之后走去深度为j的祖先节点的“XXXX”的最小花费
那么我们再考虑一下:我们这个点的祖先节点可能已经被点亮,也可能没有被点亮。如果已经点亮,我们就必须去点亮祖先的兄弟那棵子树。
因此我们定义两个数组:
f(ather)[i][j]表示走完以i为根的子树之后去点亮深度为j的祖先节点的最小花费
b(rother)[i][j]表示走完以i为根的子树之后去点亮深度为j的祖先节点的兄弟节点的最小花费
接下来我们考虑状态的转移。最好想到的是节点i是叶子节点的情况:直接走去对应的节点即可。
完整代码见下:
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
const int N=;
typedef long long LL;
int n,l[N],r[N],deep[N];
LL f[N][],b[N][],a[N],lval[N],rval[N],dis[N];
int main()
{
scanf("%d",&n);LL v;int rt;deep[]=;
for(int i=;i<=n;i++)scanf("%lld",&a[i]);
for(int i=;i<n;i++)
{
scanf("%lld",&v);
rt=(i+)>>,deep[i+]=deep[rt]+;
if((i+)&) r[rt]=i+,rval[rt]=v,dis[r[rt]]=dis[rt]+rval[rt];
else l[rt]=i+,lval[rt]=v,dis[l[rt]]=dis[rt]+lval[rt];
}
for(int i=n;i>;i--)
for(int j=;j<=deep[i];j++)
if(!r[i])
if(!l[i])
{
int fa=i>>(deep[i]-j+),fab=(i>>(deep[i]-j))^;
b[i][j]=( dis[i]+dis[fab]-(dis[fa]<<) )*a[fab];
}
else b[i][j]=lval[i]*a[l[i]]+b[l[i]][j];
else b[i][j]=min(lval[i]*a[l[i]]+b[l[i]][deep[i]+]+b[r[i]][j],rval[i]*a[r[i]]+b[r[i]][deep[i]+]+b[l[i]][j]);
for(int i=n;i;i--)
for(int j=;j<=deep[i];j++)
{
if(!r[i])
if(!l[i])
if(!j)f[i][j]=;
else
{
int fa=i>>(deep[i]-j);
f[i][j]=(dis[i]-dis[fa])*a[fa];
}
else f[i][j]=lval[i]*a[l[i]]+f[l[i]][j];
else f[i][j]=min(lval[i]*a[l[i]]+b[l[i]][deep[i]+]+f[r[i]][j],rval[i]*a[r[i]]+b[r[i]][deep[i]+]+f[l[i]][j]);
}
LL ans=f[][];
for(int i=;i<=n;i++)
{
int u=i,bro=u^;
LL tmp=f[u][deep[u]-];
while(u>)
{
if(bro>n)tmp+=a[u>>]*(dis[u>>]-dis[u>>]);
else tmp+=a[bro]*(dis[bro]-dis[u>>])+f[bro][deep[u>>]-];
u>>=,bro=u^;
}
ans=min(ans,tmp);
}
printf("%lld\n",ans);
}
[BZOJ4446]SCoi2015 小凸玩密室 树形DP(烧脑高能预警)的更多相关文章
- BZOJ4446:[SCOI2015]小凸玩密室(树形DP)
Description 小凸和小方相约玩密室逃脱,这个密室是一棵有n个节点的完全二叉树,每个节点有一个灯泡.点亮所有灯泡即可逃出密室. 每个灯泡有个权值Ai,每条边也有个权值bi.点亮第1个灯泡不需要 ...
- LUOGU P4253 [SCOI2015]小凸玩密室(树形dp)
传送门 解题思路 玄学树形\(dp\),题目描述极其混乱...看错了两次题,设首先根据每次必须点完子树里的灯才能点别的,那么点灯情况只有两种,第一种是点到某一个祖先,第二种是点到某一个祖先的兄弟.所以 ...
- BZOJ.4446.[SCOI2015]小凸玩密室(树形DP)
BZOJ LOJ 洛谷 (下面点亮一个灯泡就说成染色了,感觉染色比较顺口... 注意完全二叉树\(\neq\)满二叉树,点亮第一个灯泡\(\neq\)第一次点亮一号灯泡,根节点应该就是\(1\)... ...
- BZOJ4446 [Scoi2015]小凸玩密室 【树形Dp】
题目 小凸和小方相约玩密室逃脱,这个密室是一棵有n个节点的完全二叉树,每个节点有一个灯泡.点亮所有灯 泡即可逃出密室.每个灯泡有个权值Ai,每条边也有个权值bi.点亮第1个灯泡不需要花费,之后每点亮4 ...
- 2019.03.26 bzoj4446: [Scoi2015]小凸玩密室(树形dp)
传送门 题意简述: 给一棵完全二叉树,有点权aia_iai和边权,每个点有一盏灯,现在要按一定要求点亮: 任意时刻点亮的灯泡必须连通 点亮一个灯泡后必须先点亮其子树 费用计算如下:点第一盏灯不要花费 ...
- BZOJ4446 SCOI2015小凸玩密室(树形dp)
设f[i][j]为由根进入遍历完i子树,最后一个到达的点是j时的最小代价,g[i][j]为由子树内任意一点开始遍历完i子树,最后一个到达的点是j时的最小代价,因为是一棵完全二叉树,状态数量是nlogn ...
- BZOJ4446: [Scoi2015]小凸玩密室
用ui,j表示走完i的子树后走到i的深度为j的祖先的兄弟的最小代价: 用vi,j表示走完i的子树后走到i的深度为j的祖先的最小代价,用u算出v. 枚举起点,计算答案. #include<bits ...
- [bzoj4446] [loj#2009] [Scoi2015] 小凸玩密室
Description 小凸和小方相约玩密室逃脱,这个密室是一棵有 \(n\) 个节点的完全二叉树,每个节点有一个灯泡.点亮所有灯泡即可逃出密室.每个灯泡有个权值 \(Ai\) ,每条边也有个权值 \ ...
- bzoj 4446: [Scoi2015]小凸玩密室
Description 小凸和小方相约玩密室逃脱,这个密室是一棵有n个节点的完全二叉树,每个节点有一个灯泡.点亮所有灯 泡即可逃出密室.每个灯泡有个权值Ai,每条边也有个权值bi.点亮第1个灯泡不需要 ...
随机推荐
- VRRP+tunnel+IP SLA+Track实现冗余切换
IP SLA(Internet Protocol Service-Level Agreement)互联网服务等级协议,本实验里通过发送测试报文,测试下一跳是否可达,结合Track实现冗余静态路由的切换 ...
- Unity — — UGUI之背包物品拖放
最新背包代码: Unity3D — — UGUI之简易背包 Unity版本:2017.3 功能:用UGUI实现简单的背包物品拖放/交换功能 一.简介 在UGUI下,物品的拖放脚本实现主要依赖于Unit ...
- 009--EXPLAIN用法和结果分析
在日常工作中,我们会有时会开慢查询去记录一些执行时间比较久的SQL语句,找出这些SQL语句并不意味着完事了,些时我们常常用到explain这个命令来查看一个这些SQL语句的执行计划,查看该SQL语句有 ...
- .NetCore下使用EF DbFirst操作MySql
新建.NetCore的控制台项目 使用Nuget安装Pomelo.entityframeworkcore.mysql 工程右键--->编辑.csproj文件,把以下内容写入到工程文件 <I ...
- CentOS 7 安装图形化界面及 Xshell 连接
CentOS 比较适合用作服务器的系统,之前用过 CentOS 6,但是在配置 Nginx 的时候,发现很多语句版本7的系统都进行了更新,而且网上针对版本7的例子会更多一下,遂将系统换成版本7. 下载 ...
- pwd命令详解
基础命令学习目录首页 原文链接:https://blog.csdn.net/gnail_oug/article/details/70664458 pwd是Print Working Directory ...
- 教你用Python解决非平衡数据问题(附代码)
本文为你分享数据挖掘中常见的非平衡数据的处理,内容涉及到非平衡数据的解决方案和原理,以及如何使用Python这个强大的工具实现平衡的转换. 后台回复“不平衡”获取数据及代码~ 前言 好久没有更新自己写 ...
- PSP Daily软件beta版本——基于NABCD评论,及改进建议
1.根据(不限于)NABCD评论作品的选题: 此软件的用户人群较为明确,即:用户(软件工程课上学生)记录例行报告.写每周PSP表格和统计的需求.潜在用户还有未来该课堂的学生和需要用PSP方法记录任务完 ...
- Spring笔记⑥--整合struts2
Spring如何在web应用里面用 需要额外加入的jar包 Spring-web-4.0.0 Spring-webmvc-4.0.0 Spring的配置文件,没什么不同 需要在web.xml下配置 ...
- ASP.NET中实现封装与策略模式
首先把运算方法封装起来,这样在网页界面中直接就可以调用了,不过是换张脸而已! using System; using System.Collections.Generic; using System. ...