【c++回顾】3.1经典算法问题-多重背包问题

问题描述:给定n种物品和一个背包。物品i的重量是wi,其价值为vi,背包的容量为C。应该如何选择装入背包中的物品,使得装入背包中物品的总价值最大?每种物品有ni个。
问题分析:多重背包问题就是在完全背包问题的基础上增加了物品个数的限制。按照我们原来的解题思路:“首先递归遍历出所有使背包饱和(即背包中无法再放下任何一个物品)的方案,再求取背包最大价值及最佳拿取方案。”,我们首先应求出所有使背包饱和的方案。上上篇博客中我们已经求出了物品无数量限制条件下的所有背包饱和拿取方案,而现在物品增加了数量限制ni,那么我们现在要求的解一定是无数量限制的解的子集。
具体实现上,我们只需要对递归返回的条件等进行约束,即可解决多重背包问题。
直接上代码:

//给定n种物品和一个背包。物品i的重量是wi,其价值为vi,背包的容量为C。应该如何选择装入背包中的物品,使得装入背包中物品的总价值最大?每种物品有ni个
//利用完全背包填满问题的递归函数,遍历得到所有使背包饱和的结果,求取每个方案的价值,查找最大值并输出最佳拿取方案

#include "pch.h"
#include 
#include 
using namespace std;
//int型取最小值
int Min_int(int a, int b)
{
	return a > b ? b : a;
}
//int型取最大值
int Max_int(int a, int b)
{
	return a > b ? a : b;
}
//打印拿取方案
void PrintVector(vector take_or_not)
{
	cout << "【最佳拿取方案】" << endl;
	for (int i = 0; i < take_or_not.size(); i++)
	{
		cout << "拿取" << "第" << i << "个物品:" << take_or_not[i] << " 个" << endl;
	}
	cout << endl;
}
//给出拿取情况向量与重量列表向量,求解当前背包重量,也可以用于求解价值
int CalculateWeight(vector taken_items, vector weights)
{
	int weight = 0;
	for (int i = 0; i < taken_items.size(); i++)
		weight = weight + taken_items[i] * weights[i];
	return weight;
}
//判断剩余空间能否放下从i开始的某个物品,一个都放不了则返回False
bool TakeMore(vector weights, vector number_i, vector take_or_not, int left_capacity, int i)
{
	for (; i < weights.size(); i++)
	{
		if (weights[i] <= left_capacity && take_or_not[i] < number_i[i])
			return true;
	}
	return false;
}
/*递归解决该问题*/
//给出背包(剩余)容量,物品重量列表,拿取方案列表,要拿取的物品位置i
//主函数调用时先初始化take_or_not与i为0
//该问题中,solutions用于存储所有使背包饱和的拿取方案
//不同于完全背包问题,多重背包问题中还需要增加每个物品只有n个的限制条件
void Recursion(int capacity, vector weights, vector number_i, vector& take_or_not, int i, vector>& solutions)
{
	int left_capacity = capacity - CalculateWeight(take_or_not, weights);
	//如果已经到最后一个物品,则直接用最后一个物品装满背包,进行操作,并返回
	if (i == (take_or_not.size() - 1))
	{
		//最后一个物品尽可能多拿
		take_or_not[i] = Min_int(left_capacity / weights[i], number_i[i]);
		//更新left_capacity
		left_capacity = capacity - CalculateWeight(take_or_not, weights);
		if (take_or_not[i] >= 0)
		{
			//判断当前拿取方案是否饱和,如果每个物品都放不下了再进行操作
			if (!TakeMore(weights, number_i, take_or_not, left_capacity, 0))
			{
				//operation,存储当前拿取方案
				solutions.push_back(take_or_not);
			}
		}
		//返回前将最后一个物品清空
		take_or_not[i] = 0;
		return;
	}
	//定义flag,用于判断是否执行了for循环
	bool flag = false;
	//第i个物品数量从0开始递增,直到装满背包或物品用完为止,每加1就对i+1进行递归
	for (int o = 0; weights[i] <= left_capacity && o <= number_i[i]; o++)
	{
		take_or_not[i] = o;
		//更新left_capacity
		left_capacity = capacity - CalculateWeight(take_or_not, weights);
		Recursion(capacity, weights, number_i, take_or_not, i + 1, solutions);
		flag = true;
	}
	//如果当前i不是最后一个物品,且未执行for循环,则继续递归
	if (i != take_or_not.size() - 1 && !flag)
		Recursion(capacity, weights, number_i, take_or_not, i + 1, solutions);
	//当前物品遍历结束后,该物品数量清零并返回
	take_or_not[i] = 0;
	return;
}
//在solution中查找并输出背包价值最大值及最佳拿取方案
void Backpack(vector> solutions, vector values)
{
	//定义最大价值,初始化为0
	int max_value = 0;
	//第一次遍历,求得最大值,并输出
	for (auto solution : solutions)
	{
		int current_value = CalculateWeight(solution, values);
		max_value = max_value > current_value ? max_value : current_value;
	}
	cout << "【背包最大价值】" << endl << max_value << endl;
	//第二次遍历,求得最佳拿取方案,并输出
	for (auto solution : solutions)
	{
		int current_value = CalculateWeight(solution, values);
		if (current_value == max_value)
			PrintVector(solution);
	}
}
int main()
{
	//初始化
	int capacity;
	vector weights;
	vector values;
	vector number_i;
	int number;
	cout << "输入背包容量及物品件数:" << endl;
	cin >> capacity >> number;
	cout << "输入这" << number << "个物品的重量" << endl;
	for (int i = 0; i < number; i++)
	{
		int weight;
		cin >> weight;
		weights.push_back(weight);
	}
	cout << "输入这" << number << "个物品的价值" << endl;
	for (int i = 0; i < number; i++)
	{
		int value;
		cin >> value;
		values.push_back(value);
	}
	cout << "输入这" << number << "个物品的个数" << endl;
	for (int i = 0; i < number; i++)
	{
		int n;
		cin >> n;
		number_i.push_back(n);
	}
	//定义拿取方案,用于递归
	vector take_or_not(number, 0);
	//定义递归操作符
	int i = 0;
	//定义solutions,用于存储所有使背包饱和的拿取方案
	vector> solutions;
	//递归获取所有solution
	Recursion(capacity, weights, number_i, take_or_not, i, solutions);
	//在solution中查找并输出背包价值最大值及最佳拿取方案
	Backpack(solutions, values);

	return 0;
}
/*
输入:
10 3
1 5 8
1 4 7
9 2 1
*/

重要的修改发生在:

  1. 第37行,TakeMore函数的修改
  2. 第58行,最后一个物品最大拿取数量的修改
  3. 第77行,某物品拿去个数 循环条件的修改

其他部分与完全背包问题几乎完全相同。

示例运行结果:
【c++回顾】3.1经典算法问题-多重背包问题_第1张图片

你可能感兴趣的:(c++回顾)