c语言背包问题(动态规划解法)

c语言背包问题(动态规划解法)题目描述:有若干个物品要装进背包,并且每个物品有各自的价值,物品的数量、价值以及背包的容量由用户输入,求背包内能够存入的最大价值为多少,并且求出此时放入了哪些物品输入格式:第一行输入物品的容量r和物品个数n第二行输入每个物品的重量第三行输入每个物品的价值输出格式:第一行输出背包中能够存储的最大价值第二行输出此时背包中的物品编号思路分析:可以把这个问题看成是一个二维数组,行是物品编号,列是背包容量,若物品编号为2,背包容量为4,代表的则是当背包容量为..

大家好,又见面了,我是你们的朋友全栈君。

题目描述:

有若干个物品要装进背包,并且每个物品有各自的价值,物品的数量、价值以及背包的容量由用户输入,求背包内能够存入的最大价值为多少,并且求出此时放入了哪些物品

输入格式:

第一行输入物品的容量r和物品个数n

第二行输入每个物品的重量

第三行输入每个物品的价值

输出格式:

第一行输出背包中能够存储的最大价值

第二行输出此时背包中的物品编号

思路分析: 

  可以把这个问题看成是一个二维数组,行是物品编号,列是背包容量,若物品编号为2,背包容量为4,代表的则是当背包容量为4的时候,前两个物品的最大价值。因此当行为物品数,列为背包容量时,即容量为n的背包能够存储的最大价值。

  因此我们定义一个函数给全局变量二维数组赋值,返回二维数组右下角的值即可。

  那么怎么判断放入了哪些物品呢,此时可以根据算法来逆推,若当前物品在当前容量时的价值,与前一个物品在相同容量时的价值相等,则证明当前物品没有被放进背包。若不相等,则证明当前物品放进了背包,此时物品数-1,容量数减去当前物品的重量。

  因此,我们再定义一个函数来寻找背包内放入了哪些物品,并且还要定义一个全局数组,数组的长度就是物品数,数组里面默认都是0,如果在函数中判断放入了该物品,则物品编号对应的值赋1,最后在主函数中判断即可。,

样例输入:

10 3
3 4 5
4 5 6

样例输出: 

11
2 3

代码实现: 

#include<stdio.h>
int f[100][100];//构建最优矩阵,定义全局变量,默认赋0
int find[100];//定义一个数组,寻找背包内放入了哪些物品

int best(int r,int n,int *weight,int *value)//定义一个构建最优矩阵的函数
{     int i,j;

      for(i=0;i<=n;i++)
        f[i][0]=0;

        for(i=0;i<=r;i++)
        f[0][i]=0;

        for(i=1;i<=n;i++)
          for(j=1;j<=r;j++)
            {   if(weight[i]<=j&&f[i-1][j-weight[i]]+value[i]>f[i-1][j])
                   f[i][j]=f[i-1][j-weight[i]]+value[i];
                 else
                 f[i][j]=f[i-1][j];
            }
            return f[n][r];

}

void object(int r,int n,int *weight)//定义一个函数寻找背包里面放入了哪些物品,因为改变的是全局变量find数组的值,所以函数可以不必有返回值
{        int i,j=r;
        for(i=n;i>=1;i--)
          {   if(f[i][j]!=f[i-1][j])  //如果当前价值,比不放当前物品的价值不一样,则代表放入了当前物品
                {find[i]=1;j=j-weight[i];}
          }
}

int main()
{  int weight[100]={0};
   int value[100]={0};
   int r,n,i,maxvalue;//r为背包容量 n为物品数量 maxvalue为最大的背包价值
   scanf("%d%d",&r,&n);

   for(i=1;i<=n;i++)
     scanf("%d",&weight[i]);//输入物品的重量

    for(i=1;i<=n;i++)
      scanf("%d",&value[i]);//输入物品的价值

      maxvalue=best(r,n,weight,value);

    object(r,n,weight);

    printf("背包的最大价值为:%d\n",maxvalue);

    printf("背包里面存入了物品:");

   for(i=1;i<=n;i++)
     if(find[i]==1)printf("%d ",i);

     return 0;

}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/159097.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • PHOTOSHOP MAC快捷键

    PHOTOSHOP MAC快捷键工具箱(多种工具共用一个快捷键的可同时按【Shift】加此快捷键选取)矩形、椭圆选框工具【M】裁剪工具【C】移动工具【V】套索、多边形套索、磁性套索【L】魔棒工具【W】喷枪工具【J】画笔工具【B】像皮图章、图案图章【S】历史记录画笔工具【Y】像皮擦工具【E】铅笔、直线工具【N】模糊、锐化、涂抹工具【R】减淡、加深、海棉工

  • 为什么补码是按位取反加一_补码为什么加1

    为什么补码是按位取反加一_补码为什么加1首先,阅读这篇文章的你,肯定是一个在网上已经纠结了很久的读者,因为你查阅了所有你能查到的资料,然后他们都会很耐心的告诉你,补码:就是按位取反,然后加一。准确无误,毫无破绽。但是,你搜遍了所有俯拾即是而且准确无误的答案,却仍然选择来看这篇毫不起眼的文章,原因只有一个,只因为你还没有得到你想要的东西。            因为你想要的,不是1+1=2,而是,1+1为什么等于2。当然,我们不讨论

  • Java经典设计模式之十一种行为型模式(附实例和详解)

    Java经典设计模式之十一种行为型模式(附实例和详解)

    2020年11月12日
  • eclipse汉化教程及汉化包

    eclipse汉化教程及汉化包https://download.eclipse.org/technology/babel/update-site/R0.18.3/2021-03/

  • object中提供的clone默认为浅克隆_接口的定义形式

    object中提供的clone默认为浅克隆_接口的定义形式cloneable接口的作用cloneable其实就是一个标记接口,只有实现这个接口后,然后在类中重写Object中的clone方法,然后通过类调用clone方法才能克隆成功,如果不实现这个接口,则会抛出CloneNotSupportedException(克隆不被支持)异常。Object中clone方法:这里有一个疑问,Object中的clone方法是一个空的方法,那么他是如何判断类…

    2022年10月15日
  • SQL Server 2008 示例库 AdventureWorks2008R2[通俗易懂]

    SQL Server 2008 示例库 AdventureWorks2008R2[通俗易懂]求AdventureWorks2008R2和AdventureWorksDW2008R2的下载地址图片:

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号