分析
选择物品放入背包时每个物品只有两种选择情况即放入与不放入,所以在n个物品的情况下采用最暴力手段解决时最坏时间复杂度应为2的n次方。
解决
假如物品数量为3,编号分别为1、2、3则情况无非如下8种(0表示不放入背包,1表示放入背包):
1 0 0 0 *
2 1 0 0 *
3 0 1 0
4 0 0 1
5 1 1 0 *
6 1 0 1
7 0 1 1
8 1 1 1 *
第1行为三个0的全排列,所以只有1种情况; 第2~4行为仅含有一个1的全排列; 第5~7行为含有两个1的全排列; 第8行为三个1的全排列。
由此,很自然的想到对深度递归全排列的方法进行升级改进以求解。
不过我总感觉应该有更简洁更直观的暴力手段来求解背包问题,而且总感觉上面写的分析和下面的解决好像其实不是同一个方法,但是自己又说不出个所以然来。 唉,没办法,自己实在是太菜了,就希望先作为随笔记录下来期待未来有一天待自己强大了再回头来看看吧。加油!!!
代码如下:
#include<bits/stdc++.h>
using namespace std;
const int N = 510;
int n, V;
int e[N], v[N], w[N], path[N];
int st[N], res = 0;
void dfs(int u){
if(u == n + 1){
int sum_v = 0, sum_w = 0;
for(int i=1; i<=n; i++){
sum_v += path[i] * v[i];
sum_w += path[i] * w[i];
// printf("%d ", path[i]);
}
puts("");
if(sum_v <= V) res = max(sum_w, res);
return;
}
for(int i=1; i<=n; i++){
if(!st[i]){
path[u] = e[i];
st[i] = true;
dfs(u + 1);
st[i] = false;
}
}
}
int main(){
freopen("0-1背包问题.txt", "r", stdin);
scanf("%d%d", &n, &V);
for(int i=1; i<=n; i++){
scanf("%d%d", v+i, w+i);
}
for(int i=1; i<=n; i++){
e[i] = 1;
dfs(1);
}
cout << res << endl;
return 0;
}
/*
测试样例
输入:
4 5
1 2
2 4
3 4
4 5
输出:
8
*/
——/嵌入式第2阶段/ ├──1.1.ARM那些你得知道的事儿-ARM裸机开篇部分(ID-3584).7z 1.22G ├──1.10.SD卡启动详解-ARM裸机第十部分(ID-4291).7z 945.56M ├──1.11.NandFlash和iNand-ARM裸机第十一部分(ID-4355).7z 1.16G ├──1.12.I2C通信详解-ARM裸机第十二部分(ID-4379).7z 923.58M ├──1.13.ADC-ARM裸机第十三部分(ID-4427).7z 596.84M ├──1.14.LCD显示器实战-ARM裸机第十四部分(ID-4472).7z 1.74G ├──1.15.触摸屏TouchScreen-ARM裸机第十五部分(ID-4485).7z 556.16M ├──1.16.shell原理和问答机制引入-ARM裸机第十六部分(ID-4498).7z 1.60G ├──1.2.ARM体系结构与汇编指令-ARM裸机第二部分(ID-3635).7z 2.53G ├──1.3.开发板、原理图和数据手册-.ARM裸机第三部分(ID-3727).7z 1.44G ├──1.4.GPIO和LED-ARM裸机第四部分视频课程(ID-3767).7z 1.90G ├──1.5.SDRAM和重定位relocate-ARM裸机第五部分(ID-3860).7z 1.92G ├──1.6.S5PV210的时钟系统-ARM裸机第六部分(ID-4092).7z 973.14M ├──1.7.串口通信详解-ARM裸机第七部分(ID-4135).7z 1.74G ├──1.8.按键和CPU的中断系统-ARM裸机第八部分(ID-4167).7z 1.66G └──1.9.定时器、看门狗和RTC-ARM裸机第九部分(ID-4235).7z 1.31G
下载地址:https://www.feimaoke.com/9851.html
|