[bzoj2427]P2515 [HAOI2010]软件安装(树上背包)
tarjan+树上背包
题目描述
现在我们的手头有 \(N\) 个软件,对于一个软件 \(i\),它要占用 \(W_i\) 的磁盘空间,它的价值为 \(V_i\)。我们希望从中选择一些软件安装到一台磁盘容量为 \(M\) 计算机上,使得这些软件的价值尽可能大(即 \(V_i\) 的和最大)。
但是现在有个问题:软件之间存在依赖关系,即软件 \(i\)只有在安装了软件 \(j\)(包括软件 \(j\) 的直接或间接依赖)的情况下才能正确工作(软件 \(i\) 依赖软件 \(j\) )。
幸运的是,一个软件最多依赖另外一个软件。如果一个软件不能正常工作,那么它能够发挥的作用为 \(0\)。
我们现在知道了软件之间的依赖关系:软件 \(i\) 依赖软件 \(D_i\)。
现在请你设计出一种方案,安装价值尽量大的软件。一个软件只能被安装一次,如果一个软件没有依赖则 \(D_i=0\),这时只要这个软件安装了,它就能正常工作。
把依赖关系想象成有向边,由被依赖的软件指向依赖它的软件
那么一个点能被选到的条件就是,它的祖先被选
然后每个点都只有一个入度,还是有向边,所以如果没有环的话,这就是一个树!直接树上背包就能行了
那么考虑用 tarjan 缩点,每个强连通分量中,每两个点都可以互相到达,而这个“互相到达”,放在这个题里就是互相直接或间接的依赖
所以强连通分量里的点,如果选就都选,不选就都不选,这点很好理解
那么缩点后的图就是树了吗?
是的,对于一个强连通分量来讲,很显然,想要满足每两个点互相到达的要求,每个点都必须有它所在的这个强连通分量中,其它的点连过来的边(废话),那么,此时它的入度已经为一了,就不会再有这个强连通分量以外的点向这个点连边了
所以,只有可能是这个强连通分量向外连边,而且是连到那种只有一个点的强连通分量中
此时要把图缩完点的情况,每个强连通分量作为节点,重新连边
那么同时构建一个虚拟节点,由它向没有入度的强联通分量连边,这个虚拟点的 \(v,w\) 均为 \(0\)
直接以这个虚拟节点为根做树上背包dp 就行了
简单说一下树上背包咋做
\(f_{u,i}\) 表示在 \(u\) 这个点,用 \(i\) 的硬盘限制,能获得的最大价值
初始是 \(f_{u,i}=v_u,i\in [w_u,m]\)
再分别枚举 \(j\in [0,m-w_u]\) 表示对 所有 子树的限制,\(k\in[0,j]\) 表示对 当前 子树的限制
\]
这个转移方程也就很好理解了
另外这个 \(0-w_u<0\) 是有可能的,然后我用 ~j 来判断它是不是等于 \(-1\) 来结束循环节爆炸了
我再也不用位运算了
#include<cstdio>
#include<algorithm>
#include<iostream>
#include<cmath>
#include<map>
#include<iomanip>
#include<cstring>
#define reg register
#define EN std::puts("")
#define LL long long
inline int read(){
register int x=0;register int y=1;
register char c=std::getchar();
while(c<'0'||c>'9'){if(c=='-') y=0;c=std::getchar();}
while(c>='0'&&c<='9'){x=x*10+(c^48);c=std::getchar();}
return y?x:-x;
}
#define N 106
#define M 506
int fir[N],nex[N],to[N],tot;
int fir_[N],nex_[N],to_[N],tot_;
int dfn[N],low[N],dfscnt;
int scc[N],scccnt,indeg[N],sum_w[N],sum_v[N];
int stack[N],top;
int val[N],w[N];
int n,m;
int f[N][M];
inline void add(int u,int v){
to[++tot]=v;
nex[tot]=fir[u];fir[u]=tot;
}
inline void add_(int u,int v){
to_[++tot_]=v;
nex_[tot_]=fir_[u];fir_[u]=tot_;
}
void tarjan(int u){
stack[top++]=u;low[u]=dfn[u]=++dfscnt;
for(reg int v,i=fir[u];i;i=nex[i]){
v=to[i];
if(!dfn[v]){
tarjan(v);
low[u]=std::min(low[u],low[v]);
}
else if(!scc[v]) low[u]=std::min(low[u],dfn[v]);
}
if(dfn[u]==low[u]){
scccnt++;
do{
scc[stack[--top]]=scccnt;
sum_w[scccnt]+=w[stack[top]];sum_v[scccnt]+=val[stack[top]];
}while(stack[top]!=u);
}
}
inline void rebuild(){
for(reg int i=1;i<=n;i++){
for(reg int j=fir[i];j;j=nex[j])if(scc[i]!=scc[to[j]])
add_(scc[i],scc[to[j]]),indeg[scc[to[j]]]=1;
}
for(reg int i=1;i<=scccnt;i++)if(!indeg[i]) add_(scccnt+1,i);
}
void dfs(int u){
for(reg int i=sum_w[u];i<=m;i++) f[u][i]=sum_v[u];
reg int v,mm;
for(reg int i=fir_[u];i;i=nex_[i]){
v=to_[i];mm=m-sum_w[u];
dfs(v);
for(reg int j=mm;j>=0;j--){//j对 所有 子树的限制
for(reg int k=0;k<=j;k++)//k是对 当前 子树的限制
f[u][j+sum_w[u]]=std::max(f[u][j+sum_w[u]],f[v][k]+f[u][j+sum_w[u]-k]);
}
}
}
int main(){
n=read();m=read();
for(reg int i=1;i<=n;i++) w[i]=read();
for(reg int i=1;i<=n;i++) val[i]=read();
for(reg int i=1,x;i<=n;i++){
x=read();
if(x) add(x,i);
}
for(reg int i=1;i<=n;i++)if(!dfn[i]) tarjan(i);
rebuild();
// EN;
// for(reg int i=1;i<=scccnt;i++) std::printf("%d : %d ",i,fir_[i]);EN;
// std::puts("new : ");
// for(reg int i=1;i<=scccnt;i++){
// std::printf("%d : ",i);
// for(reg int j=fir_[i];j;j=nex_[j]) std::printf("%d ",to_[j]);
// EN;
// }
// for(reg int i=1;i<=scccnt;i++) std::printf("%d %d\n",sum_v[i],sum_w[i]);
dfs(scccnt+1);
std::printf("%d",f[scccnt+1][m]);
return 0;
}
[bzoj2427]P2515 [HAOI2010]软件安装(树上背包)的更多相关文章
- 【BZOJ2427】[HAOI2010]软件安装 Tarjan+树形背包
[BZOJ2427][HAOI2010]软件安装 Description 现在我们的手头有N个软件,对于一个软件i,它要占用Wi的磁盘空间,它的价值为Vi.我们希望从中选择一些软件安装到一台磁盘容量为 ...
- 【BZOJ2427】[HAOI2010]软件安装(动态规划,Tarjan)
[BZOJ2427][HAOI2010]软件安装(动态规划,Tarjan) 题面 BZOJ 洛谷 题解 看到这类题目就应该要意识到依赖关系显然是可以成环的. 注意到这样一个性质,依赖关系最多只有一个, ...
- 洛谷 P2515 [HAOI2010]软件安装 解题报告
P2515 [HAOI2010]软件安装 题目描述 现在我们的手头有\(N\)个软件,对于一个软件\(i\),它要占用\(W_i\)的磁盘空间,它的价值为\(V_i\).我们希望从中选择一些软件安装到 ...
- luogu P2515 [HAOI2010]软件安装 |Tarjan+树上背包
题目描述 现在我们的手头有N个软件,对于一个软件i,它要占用Wi的磁盘空间,它的价值为Vi.我们希望从中选择一些软件安装到一台磁盘容量为MM计算机上,使得这些软件的价值尽可能大(即Vi的和最大). 但 ...
- 【bzoj2427】[HAOI2010]软件安装 Tarjan+树形背包dp
题目描述 现在我们的手头有N个软件,对于一个软件i,它要占用Wi的磁盘空间,它的价值为Vi.我们希望从中选择一些软件安装到一台磁盘容量为M计算机上,使得这些软件的价值尽可能大(即Vi的和最大).但是现 ...
- BZOJ2427:[HAOI2010]软件安装——题解
https://www.lydsy.com/JudgeOnline/problem.php?id=2427 https://www.luogu.org/problemnew/show/P2515 现在 ...
- 【BZOJ2427】[HAOI2010] 软件安装(缩点+树形DP)
点此看题面 大致题意: 有\(N\)个软件,每个软件有至多一个依赖以及一个所占空间大小\(W_i\),只有当一个软件的直接依赖和所有的间接依赖都安装了,它才能正常工作并造成\(V_i\)的价值.求在容 ...
- 洛谷 P2515 [HAOI2010]软件安装
题目描述 现在我们的手头有N个软件,对于一个软件i,它要占用Wi的磁盘空间,它的价值为Vi.我们希望从中选择一些软件安装到一台磁盘容量为M计算机上,使得这些软件的价值尽可能大(即Vi的和最大). 但是 ...
- 洛谷—— P2515 [HAOI2010]软件安装
题目描述 现在我们的手头有N个软件,对于一个软件i,它要占用Wi的磁盘空间,它的价值为Vi.我们希望从中选择一些软件安装到一台磁盘容量为M计算机上,使得这些软件的价值尽可能大(即Vi的和最大). 但是 ...
随机推荐
- 汇编刷题:根据公式 Z=(X+Y)*8-X)/4 计算Z的结果(本题将结果放在AL寄存器里)
DATA SEGMENT X DB 10 Y DB 20 Z DB 00 DATA ENDS CODE SEGMENT ASSUME CS:CODE,DS:DATA START: MOV AX,DAT ...
- 运行jmeter.bat时 提示 not able to find java executable or version
安装过好几次,这是第一次遇到运行jmeter.bat时 提示 not able to find java executable or version Please check your Java in ...
- Python-气象-大气科学-可视化绘图系列(二)——利用basemap叠加地图,并添加白化效果(代码+示例)
本文为原创链接: https:////www.cnblogs.com/zhanling/p/12193031.html 白化单图代码: import numpy as np import xarray ...
- D - Complete Tripartite
三分图染色 链接:https://codeforces.com/contest/1228/problem/D 三分图染色步骤:First 首先找一个点1作为集合A中的点,再找到与1相连的一个点设为2, ...
- SQL入门,就这么简单
随着时代的发展,人类活动产生的信息越来越多,大家常说,现在这个时代是大数据时代.在这样一个前提下,数据的存储成为我们必须要认真对待和研究的问题了.SQL(Structured Query Langua ...
- Flutter Weekly Issue 52
教程 一个易迁移.兼容性高的 Flutter 富文本方案 复杂业务如何保证Flutter的高性能高流畅度? 插件 flutter_color_models A wrapper for the Dart ...
- mongo基础
以下如有任何问题,直接到官方操作文档左上角搜索框搜索 安装 On Windows, this path is on the drive from which you start MongoDB. Fo ...
- [YII2] 增删改查2
一.新增 使用model::save()操作进行新增数据 $user= new User; $user->username =$username; $user->password =$pa ...
- SQL Server 之T-SQL基本语句 (1)
花了一天的时间看完了一本<SQL必知必会>,举个范例,来总结一下零碎的知识点.一般关于数据库操作的项目都会涉及到数据库的基本查询语句.在这里面就主要讲解一些基本常用的sql使用方法. 注: ...
- 聊聊JavaScript在工作中常用的方法(一)
一.字符串转数组(split方法) 废话少说,直接上代码: //例子1 var str="abc,def,ghi"; var strArray=str.split(",& ...