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 ...
随机推荐
- HDU 3681
也算难题,难在如何处理有些点可以无限次经过 问题. 这道题,其实很容易想到二分+TSP的状态压缩,但在处理上述问题时,确实没想到.题解是处理每一个Y或G或F点到其他YGF点的距离,BFS,这样就出现一 ...
- MySql免安装版l配置方法
初次接触mysql,折腾了一天,总是安装不成功,服务启动不了.后来从官网下载了ZIP Archive版,不用安装,直接把它解压到磁盘,做一些简单的配置就可以. 软件下载地址:http://dev.my ...
- 升级到VS2013常见问题
问题1: Building an MFC project for a non-Unicode character set is deprecated 解决方法: 用于多字节字符编码 (MBCS) 的 ...
- MTK camera 闪光灯Flashlight驱动调试流程
MTK camera 闪光灯Flashlight驱动调试流程 分类: MtkDev | 作者: topicdev 相关 | 发布日期 : 2014-09-26 | 热度 : 153° ...
- 三个命令解决ASTGO服务器重启后各种问题
SSH 命令方式登录到服务器,依次执行下面三个命令. service httpd restart service mysqld restart safe_asterisk 前面两个命令提示无效,尝试从 ...
- layui富文本编译器后台获取图片路径
@RequestMapping("add") public ModelAndView add(News news){ ModelAndView mav = ne ...
- POJ 2976 裸的01分数规划
题意:给你n个数对(认为是a数组和b数组吧),从中取n-m个数对,如果选第i个数对,定义x[i]=1,求R=∑(a[i]*x[i])/∑(b[i]*x[i])取得最大值时R的值.输出R*100(保留到 ...
- PHP 二维数组排序 可以按指定 键值排序
<?php header("Content-Type:utf-8"); $arr = array( 0 => array( 'name' => '国际原油价格', ...
- 如何在linux下搭建svn服务
• 安装svn 使用命令 yum install subversion 如果提示上述错误,请以管理员身份运行 使用命令su root 再执行 yum install subversion 2,查看sv ...
- bootstrap.min.js:6 Uncaught Error: Bootstrap's JavaScript requires jQuery at bootstrap.min.js:6
自己写了个Django系统,用到了Django-bootstrap3结果在浏览器控制台发现报错:bootstrap.min.js:6 Uncaught Error: Bootstrap's JavaS ...