传送门

题意简述:

给一棵完全二叉树,有点权aia_iai​和边权,每个点有一盏灯,现在要按一定要求点亮:

  1. 任意时刻点亮的灯泡必须连通
  2. 点亮一个灯泡后必须先点亮其子树

费用计算如下:点第一盏灯不要花费,之后如果点一盏灯uuu,且上一盏点的是vvv,花费是au∗distu,va_u*dist_{u,v}au​∗distu,v​

问把所有点都点亮的最小花费。


思路:树形dpdpdp好题。

考虑到把一棵子树点亮之后要么点亮它的某个祖先,要么点亮它的某个祖先的兄弟。

记f0/1,i,jf_{0/1,i,j}f0/1,i,j​表示把iii这棵子树全部点亮之后,把iii的jjj级祖先/iii的j−1j-1j−1级祖先的兄弟 点亮的最小总花费。

然后就可以随便转移了。

最后统计答案的时候注意细节。

代码:

#include<bits/stdc++.h>
#define ri register int
#define idx(x,y) (((1<<((y)-1))<=(x))?(x)>>(y):-1)
#define lc (p<<1)
#define rc (p<<1|1)
using namespace std;
const int rlen=1<<18|1;
inline char gc(){
    static char buf[rlen],*ib,*ob;
    (ib==ob)&&(ob=(ib=buf)+fread(buf,1,rlen,stdin));
    return ib==ob?-1:*ib++;
}
inline int read(){
    int ans=0;
    char ch=gc();
    while(!isdigit(ch))ch=gc();
    while(isdigit(ch))ans=((ans<<2)+ans<<1)+(ch^48),ch=gc();
    return ans;
}
const int N=2e5+5;
typedef long long ll;
const ll inf=1e18;
int n,a[N];
ll dis[N][18],f[2][N][18],ans=inf;
int main(){
    n=read();
    for(ri i=1;i<=n;++i)a[i]=read();
    for(ri i=2;i<=n;++i){
        dis[i][1]=read();
        for(ri j=2;~idx(i,j);++j)dis[i][j]=dis[i][1]+dis[i>>1][j-1];
    }
    for(ri p=n;p;--p){
        for(ri d=1;~idx(p,d);++d){
            f[0][p][d]=f[1][p][d]=inf;
            if(lc>n){
                f[0][p][d]=dis[p][d]*a[idx(p,d)];
                f[1][p][d]=(dis[p][d]+dis[idx(p,d-1)^1][1])*a[idx(p,d-1)^1];
            }
            else if(rc>n){
                f[0][p][d]=f[0][lc][d+1]+dis[lc][1]*a[lc];
                f[1][p][d]=f[1][lc][d+1]+dis[lc][1]*a[lc];
            }
            else{
                f[0][p][d]=min(f[1][lc][1]+f[0][rc][d+1]+dis[lc][1]*a[lc],f[1][rc][1]+f[0][lc][d+1]+dis[rc][1]*a[rc]);
                f[1][p][d]=min(f[1][lc][1]+f[1][rc][d+1]+dis[lc][1]*a[lc],f[1][rc][1]+f[1][lc][d+1]+dis[rc][1]*a[rc]);
            }
        }
    }
    for(ri s=1;s<=n;++s){
        ll tmp=f[0][s][1];
        for(ri p=s>>1,pre=s;~p;p=idx(p,1),pre>>=1){
            if((pre^1)<=n)tmp+=dis[pre^1][1]*a[pre^1]+f[0][pre^1][2];
            else tmp+=dis[p][1]*a[p>>1];
        }
        ans=min(ans,tmp);
    }
    cout<<ans;
    return 0;
}


2019.03.26 bzoj4446: [Scoi2015]小凸玩密室(树形dp)的更多相关文章

  1. [BZOJ4446]SCoi2015 小凸玩密室 树形DP(烧脑高能预警)

    4446: [Scoi2015]小凸玩密室 Time Limit: 10 Sec  Memory Limit: 128 MB Description 小凸和小方相约玩密室逃脱,这个密室是一棵有n个节点 ...

  2. BZOJ4446:[SCOI2015]小凸玩密室(树形DP)

    Description 小凸和小方相约玩密室逃脱,这个密室是一棵有n个节点的完全二叉树,每个节点有一个灯泡.点亮所有灯泡即可逃出密室. 每个灯泡有个权值Ai,每条边也有个权值bi.点亮第1个灯泡不需要 ...

  3. BZOJ.4446.[SCOI2015]小凸玩密室(树形DP)

    BZOJ LOJ 洛谷 (下面点亮一个灯泡就说成染色了,感觉染色比较顺口... 注意完全二叉树\(\neq\)满二叉树,点亮第一个灯泡\(\neq\)第一次点亮一号灯泡,根节点应该就是\(1\)... ...

  4. LUOGU P4253 [SCOI2015]小凸玩密室(树形dp)

    传送门 解题思路 玄学树形\(dp\),题目描述极其混乱...看错了两次题,设首先根据每次必须点完子树里的灯才能点别的,那么点灯情况只有两种,第一种是点到某一个祖先,第二种是点到某一个祖先的兄弟.所以 ...

  5. BZOJ4446 [Scoi2015]小凸玩密室 【树形Dp】

    题目 小凸和小方相约玩密室逃脱,这个密室是一棵有n个节点的完全二叉树,每个节点有一个灯泡.点亮所有灯 泡即可逃出密室.每个灯泡有个权值Ai,每条边也有个权值bi.点亮第1个灯泡不需要花费,之后每点亮4 ...

  6. 2019.03.26 bzoj4447: [Scoi2015]小凸解密码(线段树)

    传送门 题意简述:咕咕咕 思路:考虑预处理出bbb数组,然后每次改动aaa都只会对第iii和i+1i+1i+1这两个位置产生影响,于是可以用线段树来维护bbb数组. 现在求答案的方法是断环为链,倍增整 ...

  7. BZOJ4446 SCOI2015小凸玩密室(树形dp)

    设f[i][j]为由根进入遍历完i子树,最后一个到达的点是j时的最小代价,g[i][j]为由子树内任意一点开始遍历完i子树,最后一个到达的点是j时的最小代价,因为是一棵完全二叉树,状态数量是nlogn ...

  8. BZOJ4446: [Scoi2015]小凸玩密室

    用ui,j表示走完i的子树后走到i的深度为j的祖先的兄弟的最小代价: 用vi,j表示走完i的子树后走到i的深度为j的祖先的最小代价,用u算出v. 枚举起点,计算答案. #include<bits ...

  9. [bzoj4446] [loj#2009] [Scoi2015] 小凸玩密室

    Description 小凸和小方相约玩密室逃脱,这个密室是一棵有 \(n\) 个节点的完全二叉树,每个节点有一个灯泡.点亮所有灯泡即可逃出密室.每个灯泡有个权值 \(Ai\) ,每条边也有个权值 \ ...

随机推荐

  1. EL表达式与JSTL标签map遍历varStatus属性下标使用

    在JSP页面开发中,JSTL标签库迭代标签<c:forEach>为我们迭代遍历数组集合提供了一种选择. 遍历过程中varStatus属性为我们遍历集合提升了很大操作空间. 贴一下具体使用 ...

  2. SQL Server中与IO相关的等待类型:IO_COMPLETION和PAGEIOLATCH_*

    一个大的SQL语句操作,执行计划中包含了一个merge join操作,观察到SQL长时间处于IO_COMPLETION等待状态,如果是读取相关的表的数据,服务器应该全力为其服务,但是服务器的物理IO又 ...

  3. 吴裕雄 python深度学习与实践(18)

    # coding: utf-8 import time import numpy as np import tensorflow as tf import _pickle as pickle impo ...

  4. windows安装tf

    https://www.cnblogs.com/lvsling/p/8672404.html

  5. Unity3D人脸建模 AvataSDK研究

    1.Unity与windows交互 调用文件浏览器 1.用C#调用comdlg32.dll  ,  利用GetOpenFileName实现打开文件对话框 <1> 整体参考https://w ...

  6. Linux系统常见的压缩与打包命令

    常见的压缩文件扩展名 1.*.Z          compress程序压缩的文件 2.*.gz         gzip程序压缩的文件 3..bz2        bzip2程序压缩的文件 4..t ...

  7. Java简单操作Zookeeper

    Zookeeper客户端链接 <dependency> <groupId>org.apache.zookeeper</groupId> <artifactId ...

  8. HC-05蓝牙模块配对步骤

    参考:https://blog.csdn.net/m0_37182543/article/details/76383247

  9. jq动画设置图片抽奖

    (因为自己是前端刚入门的小白所以中间出现很多问题,不过这对于我来说就是一次经验的积累) 预想效果:点击"开始",图片循环旋转,不是同时的效果,而是有一定的时间差.点击"开 ...

  10. Vue如何更新子组件

    黑科技: 给子组件绑定一个key值,当key值改变时,子组件就会更新 ~