当前位置: 首页 > news >正文

郑州天梯网站制作seoul national university

郑州天梯网站制作,seoul national university,ui设计与艺术设计区别,郑州建站软件0/1背包问题 1.二维数组解法 题目描述:有一个容量为m的背包,还有n个物品,他们的重量分别为w1、w2、w3.....wn,他们的价值分别为v1、v2、v3......vn。每个物品只能使用一次,求可以放进背包物品的最大价值。 输入样例…

0/1背包问题

1.二维数组解法

题目描述:有一个容量为m的背包,还有n个物品,他们的重量分别为w1、w2、w3.....wn,他们的价值分别为v1、v2、v3......vn。每个物品只能使用一次,求可以放进背包物品的最大价值。

输入样例:

10 4

2 1

3 3

4 5

7 9

输出样例:

12

解:

符号描述:i表示第i个物品,背包容量为j,dp[i][j]表示从下标为0到i,背包容量为j时任意选取物品所得价值的最大值。所以全局的最优解就是dp[m][n]

背包问题和函数的递归很像,只不过函数递归时从结果去接近边界,而背包问题是从边界出发,从小问题逐步去接近最终所要求的最优解。

先创建一个二维数组

可以看到当背包容量为零,或者可选物品为0时,他的局部最优解都是0

然后从每一行的左到右开始遍历(具体是为什么可以自己试一试)

- 当背包容量为1时由于第一个物品的重量为2无法放进去,所以dp[1][1]=0;

- 当背包容量为2时可以放进第一个物品,dp[2][1]=1;

- 当背包容量大于2时,后续的最大价值都是1;

接着看第二行,这个第二行的含义就是当背包物品容量从1到j变化时,任意选物品1-2的最优解

先放的i=2的物品,然后看剩余重量能容纳的上一行的局部最优解。最后还要判断是否这个最优解比上一行同一列的最优解更大,如果更大就更新状态,否则就继承状态。

- j=1;dp[2][1]=0; 继承上一行的状态

- j=2;dp[2][2]=0; 0<1,继承上一行的状态

- j=3;dp[2][3]=3+dp[2][0]=3 ,3>1,更新状态使dp[2][3]=3;

- j=4;dp[2][4]=3+dp[2][1]=3,同样状态更新

- j=5;dp[2][5]=3+dp[1][2]=4,4>1 状态更新。

后面也是同理。

再看第三行

j从零到三无法当下第三个物品,所以此时的最优解依然是前两个物品最优选择的最优解,依旧继承上一行的状态。

然后从第4列开始,物品3就可以被放下

- j=4,dp[3][4]=5+dp[2][0]=5,5>3,状态更新

- j=5,dp[3][5]=5+dp[2][1]=5,5>4,状态更新

- j=6,dp[3][6]=5+dp[2][2]=6,6>4,状态更新

我想你聪明如你已经看到规律了,接着写出第4行

所以得出来全局的最优解就是12

下面来看代码:

#include<iostream>
#include<algorithm>using namespace std;//学校的IDE有点老,好像不支持algorithm里的max
int max(int x,int y)
{return x>y?x:y;
}int m,n;
int dp[30][30]={0};//初始化全部设置为0
int w[30];//重量
int v[30];//价值
//0/1背包问题
int main()
{cin>>m>>n;int i=0,j=0;//输入w,vfor(i=1;i<=n;i++){cin>>w[i]>>v[i];}//主要的循环体for(i=1;i<=n;i++)//物品编号遍历{for(j=1;j<=m;j++)//背包重量遍历{if(j<w[i])//这个物品无法放进去,继承上一行的状态{dp[i][j]=dp[i-1][j];}else//判断当前最优解与上一行的最优解谁更大{dp[i][j]=max(dp[i-1][j],dp[i-1][j-w[i]]+v[i]);}}}cout<<dp[n][m]<<endl;return 0;
}

2.一维数组滚动解法

我们注意到二维数组的解法的时间复杂度是m*n,空间复杂度是m*n

而暴力求解的时间复杂度是2^n,空间复杂度也是m*n

二维数组法确实优化的时间复杂度,但是空间复杂度却和暴力一样,因此便有了一维数组滚动解法来进一步优化。

我们在上面的分析中,一步步的更新局部最优解,最终得到所求的最优解。但是有时候并没有更新元素而是继承上一行的最优解,那么是不是就可以只用一个一维数组来存储第i行的最优解,然后需要更新的时候更新一下就可以了。

这时候我们就可以把原有的代码稍作修改:

#include<iostream>
#include<algorithm>using namespace std;//学校的IDE有点老,好像不支持algorithm里的max
int max(int x,int y)
{return x>y?x:y;
}int m,n;
int dp[30]={0};//初始化全部设置为0
int w[30];//重量
int v[30];//价值
//0/1背包问题
int main()
{cin>>m>>n;int i=0,j=0;//输入w,vfor(i=1;i<=n;i++){cin>>w[i]>>v[i];}//主要的循环体for(i=1;i<=n;i++){for(j=m;j>=w[i];j--)//从最右边遍历,后面的多重背包是从做左到右遍历注意区分。{dp[j]=max(dp[j],dp[j-w[i]]+v[i]);}}cout<<dp[m]<<endl;return 0;
}

电脑验证

二维数组:

完全背包问题

题目描述:有一个容量为m的背包,还有n个物品,他们的重量分别为w1、w2、w3.....wn,他们的价值分别为v1、v2、v3......vn。每个物品有无限个,求可以放进背包物品的最大价值。

输入样例:10 4 2 1 3 3 4 6 8 10

输出样例:13

完全背包区别于0/1背包就是每个物品的选择没有次数限制。

它们的解题思路的区别在于主要的循环体那,完全背包需要先继承上一层的状态,然后考虑能不能放下,如果不能那这个位置的最优解就是上层位置的最优解,否则就把这个物品放进来,再加上背包容量为j-w[i]的同层位置的最优解(同层是因为物品个数没有限制),这样就可以完成叠加。

二维数组法

来看代码:

#include<iostream>
#include<algorithm>using namespace std;int m,n;
int dp[30][30]={0};
int w[30];
int v[30];
//完全背包问题 
int main()
{int i,j;//输入m,ncin>>m>>n;// 输入w,vfor(i=1;i<=n;i++){cin>>w[i]>>v[i];} //主要循环体 for(i=1;i<=n;i++){for(j=1;j<=m;j++){//完全背包要先继承上一层状态dp[i][j]=dp[i-1][j];if(j>=w[i]){dp[i][j]=max(dp[i][j],dp[i][j-w[i]]+v[i]);} }}cout<<dp[n][m]<<endl;return 0;
}

一维数组滚动解法

这个解法同样也是为了降低空间复杂度

所以以同样的方法优化一下代码:

#include<iostream>
#include<algorithm>using namespace std;int m,n;
int dp[30]={0};
int w[30];
int v[30];
//完全背包问题 
int main()
{int i,j;//输入m,ncin>>m>>n;// 输入w,vfor(i=1;i<=n;i++){cin>>w[i]>>v[i];} //主要循环体 for(i=1;i<=n;i++){for(j=w[i];j<=m;j++){dp[j]=max(dp[j],dp[j-w[i]]+v[i]);}}cout<<dp[m]<<endl;return 0;
}

电脑验证

二维数组法:

一维数组滚动法:

多重背包问题

多重背包问题与前面两个问题的区别也是在物品的数量上,这次它换成了有限个。

题目描述:有一个容量为m的背包,还有n个物品,他们的重量分别为w1、w2、w3.....wn,他们的价值分别为v1、v2、v3......vn,它们的数量分别有c1、c2、c3......cn个,求可以放进背包物品的最大价值。

输入样例:

10 3

2 1 6

6 10 3

3 6 3

输出样例:

转换成传统的0/1背包问题

这个方法比较容易想到,就不过多赘述了

看代码:

#include<iostream>
#include<algorithm>using namespace std;int m,n;
int dp[30]={0};
int w[30];
int v[30];
int c[30];
int main()
{int i,j,k;//输入m,ncin>>m>>n;//输入w,v,c(数量) for(i=1;i<=n;i++){cin>>w[i]>>v[i]>>c[i];} for(i=1;i<=n;i++){for(k=1;k<=c[i];k++)//多次模拟0/1背包 {for(j=m;j>=w[i];j--)//一维滚动法 {dp[j]=max(dp[j],dp[j-w[i]]+v[i]);	 			}}}for(i=1;i<=m;i++)//这里直接电脑验证了 {cout<<dp[i]<<" ";}cout<<endl;cout<<dp[m];return 0;} //10 3 2 1 6 6 10 3 3 6 3

特别感谢某站T_zhao 老师的讲解,讲的很明白。

http://www.ds6.com.cn/news/93992.html

相关文章:

  • 企业简介模板100字西安网站seo哪家公司好
  • 免费空间做淘宝客网站什么关键词可以搜到那种
  • 学校如何报销网站开发费用互联网营销成功案例
  • 青秀网站建设今日新闻国际最新消息
  • 电子商城系统的设计与实现优化排名
  • phpcms v9 实现网站搜索系统优化大师下载
  • 太原百度关键词推广网站推广优化外包公司哪家好
  • 培训网站设计师营销网站建设推广
  • 动漫网站设计论文如何打百度人工电话
  • 青岛模版网站建设长春网站搭建
  • 专业做pc+手机网站企业网络营销策略分析案例
  • 成都网站建设哪家好文章金华网站建设
  • 哪些网站做推广比较有效果拼多多女装关键词排名
  • 东莞横沥理工学校吉林关键词排名优化软件
  • 在pc端网站基础上做移动端网站建设高端公司
  • 怎样给公司做网站视频互联网推广选择隐迅推
  • 郑州做营销型网站的公司西安疫情最新数据消息5分钟前
  • 骗别人做网站叶涛网站推广优化
  • 做网站多少钱PageAdminseo内部优化方案
  • 长沙房地产管理局seo网站推广企业
  • 网站程序怎么做fifa最新排名出炉
  • 网站设计价格最新疫情最新消息
  • 做网站用什么网站运营专员
  • wordpress小说网站模板网络运营推广合作
  • 网站建设演示ppt模板江苏网页设计
  • 公司网站 正式上线广告投放优化师
  • 怎样建立自己的网站赚钱安卓手机游戏优化器
  • 广州微信网站建设市场个人怎么做网站
  • 网站编辑工作媒体:北京不再公布各区疫情数据
  • 商丘家居网站建设短视频搜索优化