C語言背包問題求解全過程(貪心方法)
問題描述:
- 有一個背包,背包容量是M=150。有7個物品,物品可以分割成任意大小。要求盡可能讓裝入背包中的物品總價值最大,但不能超過總?cè)萘俊?/li>
- 物品:A B C D E F G
- 重量:35 30 60 50 40 10 25
- 價值:10 40 30 50 35 40 30
算法描述:
貪心算法(又稱貪婪算法)是指,在對問題求解時,總是做出在當前看來是最好的選擇。也就是說,不從整體最優(yōu)上加以考慮,他所做出的是在某種意義上的局部最優(yōu)解。
貪心算法不是對所有問題都能得到整體最優(yōu)解,關(guān)鍵是貪心策略的選擇,選擇的貪心策略必須具備無后效性,即某個狀態(tài)以前的過程不會影響以后的狀態(tài),只與當前狀態(tài)有關(guān)。
問題分析:
1.目標函數(shù): ∑pi最大,使得裝入背包中的所有物品pi的價值加起來最大。
2.約束條件:裝入的物品總重量不超過背包容量:∑wi<=M( M=150)
3.貪心策略: 選擇單位重量價值最大的物品
算法設(shè)計:
- 計算出每個物品單位重量的價值
- 按單位價值從大到小將物品排序
- 根據(jù)背包當前所剩容量選取物品
- 如果背包的容量大于當前物品的重量,那么就將當前物品裝進去。否則,那么就將當前物品分割再裝進去,然后跳出循環(huán)結(jié)束。
代碼實現(xiàn):
#include<bits/stdc++.h>
using namespace std;
const int maxn = 1000;
float z[maxn];
void Sort(int n,float w[],float v[]){
for(int i=0;i<n;i++)
z[i]=v[i]/w[i];//用z[]存物品的單位重量價值
for(int i=0;i<n;i++){//此排序的策略是每次把單位重量物品的價值最大的物品放在前面
for(int j=i+1;j<n;j++){
if(z[i]<z[j]){
float temp = z[i];
z[i] = z[j];
z[j]=temp;
float tempw = w[i];
w[i] = w[j];
w[j] = tempw;
float tempv = v[i];
v[i] = v[j];
v[j] = tempv;
}
}
}
}
void fire(int n,float w[],float v[],float x[],float pimax){
Sort(n,w,v);//根據(jù)單位重量物品的價值對物品進行排序
int i;
for(i=0;i<n;i++){
if(w[i]>pimax) break;
x[i] = 1; //x[]數(shù)組用來記錄此次是否選擇物品,1代表全部拿走,0代表不拿,小數(shù)代表部分拿
pimax -= w[i];
}
if(i<=n-1) x[i] = pimax/w[i];
}
int main(){
int n;
float pi=0;
float pimax,v[maxn],w[maxn],x[maxn];//w[],每個物品的重量,v[]代表每個物品的價值,pimax代表最大容量
memset(x,0,sizeof(x));
cout<<"請輸入最大容量:";
cin>>pimax;
cout<<"請輸入物品(物品可以任意分割)數(shù)量:";
cin>>n;
cout<<"請輸入每個物品的重量和價值:"<<endl;
for(int i=0;i<n;i++){
cin>>w[i]>>v[i];
}
fire(n,w,v,x,pimax);
for(int i=0;i<n;i++){
if(x[i]==1){
pi+=v[i];
}
else{
pi+=v[i]*x[i];
}
}
cout<<"最終收獲的物品(物品可以任意分割)價值為:"<<pi<<endl;
return 0;
}運行結(jié)果:

總結(jié)
到此這篇關(guān)于C語言背包問題求解(貪心方法)的文章就介紹到這了,更多相關(guān)C語言背包問題內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

