装箱问题的三种解法

有一个箱子容量为V(正整数,0≤v≤20000),同时有n个物品(0< n ≤30),每个物品有一个体积(正整数)。

要求n个物品中,任取若干个装入箱内,使箱子的剩余空间为最小。

样例
输入样例
复制代码
24
6
8
3
12
7
9
7
输出样例
复制代码
0

方法一:动规

dp[j]表示体积为j用掉的最大体积

动态转移方程:dp[j]=max(dp[j],dp[j-a[i]]+a[i]);

dp[j-a[i]]+a[i]表示a[i]全部装入

dp[j]表示不放入

cs 复制代码
#include<iostream>
#include<algorithm>
using namespace std;
int main(){
	int V,n,i,j;
	cin>>V>>n;
	int a[35]={};
	for(i=0;i<n;i++){
		cin>>a[i];
	}
	int dp[20005]={};
	for(i=0;i<n;i++){
		for(j=V;j>=a[i];j--){
			dp[j]=max(dp[j],dp[j-a[i]]+a[i]);
		}
	}
	cout<<V-dp[V]<<endl;
} 

方法2 回溯

cs 复制代码
#include<iostream>
#include<algorithm>
using namespace std;
int a[35]={}; 
//maxV表示消耗的最大体积 
int maxV=0,n,V;
//sumV为实际消耗的体积 
//cnt为搜索的第cnt个物品 
void dfs(int cnt,int sumV){
	if(sumV>V)return;
	//搜索到最后一个 
	if(cnt==n+1){
		if(sumV>maxV)maxV=sumV;
		return; 
	}
	//不放入 
	dfs(cnt+1,sumV);
	//全部放入 
	dfs(cnt+1,sumV+a[cnt]);
}
int main(){
	cin>>V;
	int i,j;
	cin>>n; 
	for(i=0;i<n;i++){
		cin>>a[i]; 
	}
	dfs(0,0);
    cout<<V-maxV<<endl;
} 

方法3 贪心

从大到小排序,每次选择可以装入的

cs 复制代码
#include<iostream>
#include<algorithm>
using namespace std;
//从大到小排序 
int cmp(const void *a,const void *b){
	return *(int*)b-*(int*)a;
} 
int main(){
	int V,n,i,j;
	cin>>V>>n;
	int a[35]={};
	for(i=0;i<n;i++){
		cin>>a[i];
	}
	qsort(a,n,sizeof(int),cmp);
	for(i=0;i<n;i++){
		if(V==0)break;
		if(V>=a[i])V-=a[i]; 
	}
	cout<<V<<endl; 
} 

这样子有局限性,只有数组最大值放入能够使剩余空间最小才满足。

举个例子

数组最大值放入反而得不到剩余空间最小值

用动态规划和回溯均可得到最小剩余空间为0。可见这个贪心策略有局限性。

相关推荐
Lips6112 小时前
2026.1.20力扣刷题笔记
笔记·算法·leetcode
2501_941329722 小时前
YOLOv8-LADH马匹检测识别算法详解与实现
算法·yolo·目标跟踪
洛生&2 小时前
Planets Queries II(倍增,基环内向森林)
算法
剑锋所指,所向披靡!2 小时前
C++之类模版
java·jvm·c++
小郭团队2 小时前
1_6_五段式SVPWM (传统算法反正切+DPWM2)算法理论与 MATLAB 实现详解
嵌入式硬件·算法·matlab·dsp开发
小郭团队2 小时前
1_7_五段式SVPWM (传统算法反正切+DPWM3)算法理论与 MATLAB 实现详解
开发语言·嵌入式硬件·算法·matlab·dsp开发
鱼跃鹰飞3 小时前
Leetcode347:前K个高频元素
数据结构·算法·leetcode·面试
bybitq3 小时前
LeetCode236-二叉树的最近公共祖先(LCA)问题详解-C++
算法·深度优先
C+-C资深大佬3 小时前
C++风格的命名转换
开发语言·c++
啊阿狸不会拉杆3 小时前
《数字图像处理》第 7 章 - 小波与多分辨率处理
图像处理·人工智能·算法·计算机视觉·数字图像处理