南京大学厦门大学ACM百练算法设计与分析OpenJudge第六章习题解答 下载本文

内容发布更新时间 : 2024/9/19 17:36:07星期一 下面是文章的全部内容请认真阅读。

第6章课后作业结题报告

第一题:

给定n种物品和一个背包,物品i(1≤i≤n)的重量是wi,其价值为vi,背包的容量为C,对每种物品只有两种选择:装入背包或者不装入背包。如何选择装入背包的物品,使得装入背包中物品的总价值最大?

这是一道老生常谈的背包问题,属于入门dp(Dynamic Programming动态规划), 首先给出状态转移方程,设dp[i][j]表示用大小为j的背包去装前i个所能获得的最大价值, w[i]表示第i个物品的体积,v[i]表示第i个物品的价值。关于第i个物品的决策,就是间单的”取与不取”, 我们很容易得到如下的状态转移方程:

dp[i][j]=max(dp[i-1][j], dp[i-1][j-w[i]]+v[i]);

这边dp[i-1][j]表示第i个物品不取,dp[i-1][j-w[i]]+v[i]表示第i个物品取

因为每次推导只用的到i-1维,我们其实可以只用一维的滚动数组就可以来求解。这时候要逆序求解,这边不多做介绍

接下来是打印路径问题,这个需要逆序贪心打印。第i个背包是否选取应该用dp[i][V]与dp[i-1][V]来比较,从而求解,dp[i][V]>dp[i-1][V],V表示当前可存在的最大容量。根据贪心,容量V越大价值同样装前i个一定更优,后面的推导的正解一定是基于dp[i][V]的,所以可以这么选

标程如下:

#include #include using namespace std; int dp[110][1010]; int n, c, ans[110]; int w[110], v[110]; void print(){ int i, j, k; k=c;

for(i=n; i>=1; i--){

if(dp[i][k]!=dp[i-1][k]){ ans[i]=1; k=k-w[i];

} else{ } }

return; }

int main(){

int i, j, k;

scanf(\ for(i=1; i<=n; i++){

scanf(\ }

for(i=1; i<=n; i++){ for(j=1; j<=c; j++){ dp[i][j]=dp[i-1][j]; if(j>=w[i]){

dp[i][j]=max(dp[i-1][j-w[i]]+v[i], dp[i][j]); } } }

printf(\ print();

for(i=1; i<=n; i++) {

printf(\ }

return 0; }

第二题:

设有n种不同面值的硬币,各硬币的面值存于数组T[1:n]中。现要用这些面值的硬币来找钱。可以使用的各种面值的硬币个数存于数组Coins[1:n]中。对任意钱数0≤m≤20001,设计一个用最少硬币找钱m的方法。

对于给定的1≤n≤10,硬币面值数组T和可以使用的各种面值的硬币个数数组 Coins,以及钱数m,0≤m≤20001,计算找钱m的最少硬币数。

这一题属于背包问题的变形

这首先转化为背包问题,如下: 如输入数据:

1 3 2 3 5 3

其实可以转化为: 体积分别为:

1 1 1 2 2 2 5 5 5, 价值都为1的物品,问题转化成问你物体的体积总和恰好为m时需要最小的价值数,

dp[i][j]表示前i个物品中抽取任意个恰好为m,所耗费的最少价值数

dp[i][j]=min(dp[i-1][j], dp[i-1][j-w[i]]+1);

但这边代码书写的时候要注意不是每个dp[i][j]的值都是可达到的,

即时任意一个数,不是你想凑就能凑出来的,所以这边要特判一下,先初始化为

足够大,如2000000000,然后在状态转移的时候都要先判断, 子问题是否已经被求解。

标程如下:

#include #include

using namespace std; int w[110], v[110]; int dp[110][20010]; int n, m; void init(){

for(int i=0; i<110; i++){

for(int j=0; j<20010; j++){ dp[i][j]=2000000000; } } }

int main(){

int i, j, k;

int t, x, cnt=1; scanf(\ for(i=1; i<=n; i++){

scanf(\ for(j=1; j<=x; j++){ w[cnt]=t;