HAOI2010软件安装(树形背包)
HAOI2010软件安装(树形背包)
题意
有n个物品,每个物品最多会依赖一个物品,但一个物品可以依赖于一个不独立(依赖于其它物品)的物品,且可能有多个物品依赖一个物品,并且依赖关系可能形成一个环。现给你V的资金,问如何分配资金,可以使你的得到的总价值最大,请求出这个总价值。
解法
我以前写过对于普通依赖性背包的博客:noip2006金明的预算方案如果对依赖性背包不是很熟悉的同学可以先看一下这道题。
由于这道题的依赖关系可能形成环,所以我们先用tarjan缩一下点,然后依赖关系图就变成了一个森林,这时候我们再将每一棵树的根节点向0号结点连一条边,表示他们依赖0号结点。这时候我们就得到了一颗依赖关系树。
那么现在的重点就在如何处理这颗依赖关系树。因为树上每一个节点的决策数都太大了,所以同样的我们考虑求出每一个以i节点为根的子树在任意权值下的最大价值,然后再在i节点利用01背包来合并,一直递归往上。最后的答案就是0号结点所有方案中最优的那一个。
ps:如果还是看不懂的话,可以参观这里和这里我也是在那儿学的。
代码
以下的代码是 \(O(nv^2)\) 的:
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <cctype>
#include <vector>
#define INF 2139062143
#define MAX 0x7ffffffffffffff
#define del(a,b) memset(a,b,sizeof(a))
using namespace std;
typedef long long ll;
template<typename T>
inline void read(T&x)
{
x=0;T k=1;char c=getchar();
while(!isdigit(c)){if(c=='-')k=-1;c=getchar();}
while(isdigit(c)){x=x*10+c-'0';c=getchar();}x*=k;
}
const int maxn=2500+15;
int w[maxn],cost[maxn];
int n,m,ecnt;
struct Edge{
int u,v;
Edge(int u,int v):u(u),v(v){}
};
vector<Edge> edge;
vector<int> G[maxn];
void add_edge(int u,int v) {
edge.push_back(Edge(u,v));
ecnt=edge.size();
G[u].push_back(ecnt-1);
}
int dfn[maxn],low[maxn],sta[maxn],top,bl[maxn],cnt;
bool ins[maxn];
void tarjian(int u) {
ins[u]=1;sta[++top]=u;
dfn[u]=low[u]=++cnt;
for(int i=0;i<G[u].size();i++) {
Edge e=edge[G[u][i]];
int v=e.v;
if(!dfn[v]) {
tarjian(v);
low[u]=min(low[u],low[v]);
}
else if(ins[v]) low[u]=min(low[u],dfn[v]);
}
if(low[u]==dfn[u]) {
int y;
while((y=sta[top--])&&y) {
bl[y]=u;
ins[y]=0;
if(u==y) break;
w[u]+=w[y],cost[u]+=cost[y];
}
}
}
int dp[maxn][maxn];
void dfs(int u) {
dp[u][cost[u]]=w[u];
for(int i=0;i<G[u].size();i++) {
Edge e=edge[G[u][i]];
int v=e.v;
dfs(v);
for(int i=m;i>=cost[u];i--)
for(int j=0;j<=i-cost[u];j++)
dp[u][i]=max(dp[u][i],dp[u][i-j]+dp[v][j]);
}
}
bool noroot[maxn];
int main()
{
read(n),read(m);
for(int i=1;i<=n;i++) read(cost[i]);
for(int i=1;i<=n;i++) read(w[i]);
for(int i=1;i<=n;i++) {
int d;read(d);
if(!d) continue;
add_edge(d,i);
}
for(int i=1;i<=n;i++) if(!dfn[i]) tarjian(i);
int k=edge.size();
for(int i=0;i<=n;i++) G[i].clear();
for(int i=0;i<k;i++) {
Edge e=edge[i];
int f1=bl[e.u],f2=bl[e.v];
if(f1==f2) continue;// 新的两个点可能在同一个强连通分量中!!
add_edge(f1,f2);noroot[f2]=1;
}
for(int i=1;i<=n;i++) if((bl[i]==i)&&(!noroot[i])) add_edge(0,i);
dfs(0);
int ans=0;
for(int i=0;i<=m;i++)
ans=max(ans,dp[0][i]);
printf("%d\n",ans);
return 0;
}
这下面的是参照徐持横的算法做到的 \(O(nv)\) 的算法:
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <cctype>
#include <vector>
#define INF 2139062143
#define MAX 0x7ffffffffffffff
#define del(a,b) memset(a,b,sizeof(a))
using namespace std;
typedef long long ll;
template<typename T>
inline void read(T&x)
{
x=0;T k=1;char c=getchar();
while(!isdigit(c)){if(c=='-')k=-1;c=getchar();}
while(isdigit(c)){x=x*10+c-'0';c=getchar();}x*=k;
}
const int maxn=500+15;
int w[maxn],cost[maxn];
int n,m,ecnt;
struct Edge{
int u,v;
Edge(int u,int v):u(u),v(v){}
};
vector<Edge> edge;
vector<int> G[maxn];
void add_edge(int u,int v) {
edge.push_back(Edge(u,v));
ecnt=edge.size();
G[u].push_back(ecnt-1);
}
int dfn[maxn],low[maxn],sta[maxn],top,bl[maxn],cnt;
bool ins[maxn];
void tarjian(int u) {
ins[u]=1;sta[++top]=u;
dfn[u]=low[u]=++cnt;
for(int i=0;i<G[u].size();i++) {
Edge e=edge[G[u][i]];
int v=e.v;
if(!dfn[v]) {
tarjian(v);
low[u]=min(low[u],low[v]);
}
else if(ins[v]) low[u]=min(low[u],dfn[v]);
}
if(low[u]==dfn[u]) {
int y;
while((y=sta[top--])&&y) {
bl[y]=u;
ins[y]=0;
if(u==y) break;
w[u]+=w[y],cost[u]+=cost[y];
}
}
}
int dp[maxn][maxn];
void dfs(int u,int M) {
if(M<=0) return;
for(int i=0;i<G[u].size();i++) {
Edge e=edge[G[u][i]];
int v=e.v;
for(int j=1;j<=M;j++) dp[v][j]=dp[u][j];
dfs(v,M-cost[v]);
for(int j=cost[v];j<=M;j++) dp[u][j]=max(dp[u][j],dp[v][j-cost[v]]+w[v]);
}
}
bool noroot[maxn];
int main()
{
read(n),read(m);
for(int i=1;i<=n;i++) read(cost[i]);
for(int i=1;i<=n;i++) read(w[i]);
for(int i=1;i<=n;i++) {
int d;read(d);
if(!d) continue;
add_edge(d,i);
}
for(int i=1;i<=n;i++) if(!dfn[i]) tarjian(i);
int k=edge.size();
for(int i=0;i<=n;i++) G[i].clear();
for(int i=0;i<k;i++) {
Edge e=edge[i];
int f1=bl[e.u],f2=bl[e.v];
if(f1==f2) continue;// 新的两个点可能在同一个强连通分量中!!
add_edge(f1,f2);noroot[f2]=1;
}
for(int i=1;i<=n;i++) if((bl[i]==i)&&(!noroot[i])) add_edge(0,i);
dfs(0,m);
int ans=0;
for(int i=0;i<=m;i++) ans=max(ans,dp[0][i]);
printf("%d",ans);
return 0;
}
HAOI2010软件安装(树形背包)的更多相关文章
- BZOJ2427:[HAOI2010]软件安装(树形DP,强连通分量)
Description 现在我们的手头有N个软件,对于一个软件i,它要占用Wi的磁盘空间,它的价值为Vi.我们希望从中选择一些软件安装到一台磁盘容量为M计算机上,使得这些软件的价值尽可能大(即Vi的和 ...
- 【BZOJ2427】[HAOI2010]软件安装 Tarjan+树形背包
[BZOJ2427][HAOI2010]软件安装 Description 现在我们的手头有N个软件,对于一个软件i,它要占用Wi的磁盘空间,它的价值为Vi.我们希望从中选择一些软件安装到一台磁盘容量为 ...
- BZOJ_2427_[HAOI2010]软件安装_tarjan+树形DP
BZOJ_2427_[HAOI2010]软件安装_tarjan+树形DP 题意: 现在我们的手头有N个软件,对于一个软件i,它要占用Wi的磁盘空间,它的价值为Vi.我们希望从中选择一些软件安装到一台磁 ...
- Tarjan+树形DP【洛谷P2515】[HAOI2010]软件安装
[洛谷P2515][HAOI2010]软件安装 题目描述 现在我们的手头有N个软件,对于一个软件i,它要占用Wi的磁盘空间,它的价值为Vi.我们希望从中选择一些软件安装到一台磁盘容量为M计算机上,使得 ...
- [HAOI2010]软件安装(Tarjan,树形dp)
[HAOI2010]软件安装 题目描述 现在我们的手头有N个软件,对于一个软件i,它要占用Wi的磁盘空间,它的价值为Vi.我们希望从中选择一些软件安装到一台磁盘容量为M计算机上,使得这些软件的价值尽可 ...
- bzoj 2427 [HAOI2010]软件安装 Tarjan缩点+树形dp
[HAOI2010]软件安装 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 2029 Solved: 811[Submit][Status][Dis ...
- 洛谷 P2515 [HAOI2010]软件安装 解题报告
P2515 [HAOI2010]软件安装 题目描述 现在我们的手头有\(N\)个软件,对于一个软件\(i\),它要占用\(W_i\)的磁盘空间,它的价值为\(V_i\).我们希望从中选择一些软件安装到 ...
- [BZOJ2427][HAOI2010]软件安装(Tarjan+DP)
2427: [HAOI2010]软件安装 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1987 Solved: 791[Submit][Statu ...
- bzoj2427:[HAOI2010]软件安装(Tarjan+tree_dp)
2427: [HAOI2010]软件安装 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1053 Solved: 424[Submit][Statu ...
随机推荐
- Android 应用启动动画代码
requestWindowFeature(Window.FEATURE_NO_TITLE);//设置无标题 setContentView(R.layout.activity_main); getWin ...
- LeetCode OJ 之 Valid Anagram
题目: Given two strings s and t, write a function to determine if t is an anagram of s. For example, s ...
- 【特征匹配】SIFT原理之KD树+BBF算法解析
转载请注明出处:http://blog.csdn.net/luoshixian099/article/details/47606159 继上一篇中已经介绍了SIFT原理与C源代码剖析,最后得到了一系列 ...
- 怎样动态设置GridView的宽和高(Android)
LinearLayout.LayoutParams linearParams2 = (LinearLayout.LayoutParams)liveGrid.getLayoutParams(); lin ...
- bzoj5441: [Ceoi2018]Cloud computing
跟着大佬做题.. 这题也是有够神仙了.观察一下性质,c很小而f是一个限制条件(然而我并不会心态爆炸) %了一发,就是把电脑和订单一起做背包,订单的c视为负而电脑的v为负,f由大到小排序做背包 #inc ...
- ASP.NET通用权限系统快速开发框架
系统在线演示地址: http://120.90.2.126:8051 登录账户:system,密码:system### DEMO下载地址: http://download.csdn.net/detai ...
- POJ - 3281 Dining(拆点+最大网络流)
Dining Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 18230 Accepted: 8132 Descripti ...
- python-day2 切片,格式化输出,函数
1.切片:取元素 格式;变量名[M:N:K] M 表示开始元素索引值, N 表示结束元素索引值(不包含索引值本身) K 表示步长,隔几个切一次 例子:a='hello python' p ...
- springboot启动报错:Cannot determine embedded database driver class for database type NONE.
package cn.zb.test; import org.springframework.boot.SpringApplication; import org.springframework.bo ...
- Blender插件加载研究
目标 [x] 解析Blender插件代码加载原理, 为测试做准备 结论 采用方法3的方式, 可以在测试中保证重新加载子模块, 是想要的方式, 代码如下: _qk_locals = locals() d ...