装箱问题

装箱问题的几种解法

  1. 暴力求解
  2. 动态规划(贪心算法)
  3. 递推式解法

题目:装箱问题
时间限制:1.0s 内存限制:256.0MB
问题描述:有一个箱子容量为V(正整数,0<=V<=20000),同时有n个物品(0<n<=30),每个物品有一个体积(正整数)。
  要求n个物品中,任取若干个装入箱内,使箱子的剩余空间为最小。
输入格式:第一行为一个整数,表示箱子容量;
  第二行为一个整数,表示有n个物品;
  接下来n行,每行一个整数表示这n个物品的各自体积。
输出格式:一个整数,表示箱子剩余空间。
样例输入
24
6
8
3
12
7
9
7
样例输出
0

1.暴力求解法

#include<iostream>
#include<algorithm>
using namespace std;
void cal(int N,int V,int v[]);
int main(){
    int V=0,N=0,*v=NULL;
    cin>>V>>N;
    v=new int[N];
    for(int i=0;i<N;i++){
        cin>>v[i];
    }
    sort(v,v+N,greater<int>());
    cal(N,V,v);
    delete []v; 
    return 0;
}
void cal(int N,int V,int v[]){
    int max[1000]={0},index=0,temp=0,flag[1000]={0},jl=0; 
    for(int i=0;i<N;i++){
        if(temp==V){
            cout<<"0";
            return;
        } 
        temp=v[i];
        if(temp>V)continue;
        for(int j=0;j<N;j++){
            flag[jl++]=j;//第一次进来是比下一个小标小一 
            if(temp<V&&v[j]<V){//如果所得的和小于V,则要继续相加 
                if(i!=j){
                    temp=temp+v[j];
                    if(temp>V){
                        max[index++]=temp-v[j];//记录每个基数和其他数相加结果的最大值并小于v 
                        if(j==N-1){
                            if(flag[0]+2!=N-1)
                                j=flag[0]+1;
                            else j=flag[0];
                            jl=0;
                            continue;
                        }   
                    }
                    if(temp<V&&j==N-1)max[index++]=temp;
                }
            }else if(temp>V){
                temp=temp-v[j-1];//如果是小于则应该减去刚加的进行下一个循环
                j=j-1;
                continue;
            }else if(temp==V){
                cout<<"0"<<endl;
                return;
            }else{
                if(temp<V&&j==N-1)max[index++]=temp;
                continue;//如果要加的这个数以后大于V 则要往后找 
            }               
        }
    }
    temp=0;
    for(int i=0;i<index;i++){
        for(int j=0;j<index;j++){
            if(max[j]<max[j+1]){
                temp=max[j];
                max[j]=max[j+1];
                max[j+1]=temp;
            }
        }
    }
    cout<<V-max[0]<<endl;
}

2.动态规划(贪心算法)
代码1:

#include<iostream>
#include<algorithm>
using namespace std;
void comp(int m[],int n,int v){
    int *t=new int[n+1];
    int *x=new int[n+1];
    sort(m,m+n);
    for(int i=1;i<=n;i++){
        x[i]=0;
        t[i-1]=i;
    }
    for(int i=1;i<=n&&m[t[i]]<=v;i++){
        x[t[i]]=1;
        v-=m[t[i]];
    }
    cout<<v;
}
int main(){
    int *m=NULL,n,v;
    cin>>v>>n;
    m=new int[n+1];
    for(int i=0;i<n;i++){
        cin>>m[i];
    }
    comp(m,n,v);     
    delete []m;
    return 0;
}

代码2:

#include <stdio.h>
#include <stdlib.h>
#define N 30

int box(int,int,int []);
int main()
{
    int Vu;
    int nu,i,m,min;
    int Vo[N];
    scanf("%d",&Vu);
    scanf("%d",&nu);
    for(i=0;i<nu;i++)
        scanf("%d",&Vo[i]);                    
    min=box(nu-1,Vu,Vo);                    
    printf("%d\n",min); 

    return 0;                   
}
int box(int n ,int V ,int a[])                        
{
    int minv,t,m=V;
    if(n==0)
    {
        if(a[n]<=V)                       
            minv=V-a[n];
        else
            minv=V;
    }
    else
    {
        t=box(n-1,V,a);
        if(a[n]<=V)                        
            m=box(n-1,V-a[n],a);      
        if(t<m)
            minv=t;
        else
            minv=m;
    }
    return minv;
}

代码3:

#include<stdio.h>
int v[30], dp[20000];
int main(){
    int n, V, i, j;
    scanf("%d", &V);
    scanf("%d", &n);
    for(i = 1; i <= n; i++)
        scanf("%d", &v[i]);
    for(i = 1; i <= n; i++){
        for(j = V; j >= v[i]; j--) 
            dp[j] = dp[j-v[i]] + v[i] > dp[j] ? dp[j-v[i]] + v[i] : dp[j];
    }
    printf("%d", V-dp[V]);
    return 0;
}

3.递推式解法

#include<iostream> 
#include<string.h>
using namespace std; 
int main(){ 
    int i,j,m,n,a;//m是箱子的容量,n是有n个物品,a是每个物品的体积 
    bool *v=NULL;
    cin>>m>>n; 
    v=new bool[m+1];
    memset(v,0,sizeof(bool)*(m+1));
    v[0]=1; 
    for(i=1;i<=n;i++){ 
        cin>>a; 
        for(j=m;j>=a;j--)
            v[j]=(v[j]||v[j-a]);//用F[i,j]表示前i个物品装j的容量是否可以,则F[i,j]=F[i-1,j-a[i]]||F[i-1,j]。其中a[i]表示第i个物品的体积 
    } 
    for(j=m;j>=0;j--) 
        if(v[j]) 
        { 
            cout<<m-j; 
            return 0; 
        } 
}

(本文完)

  • 3
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值