`
guojinhua
  • 浏览: 13694 次
  • 性别: Icon_minigender_2
  • 来自: 沈阳
最近访客 更多访客>>
社区版块
存档分类
最新评论

使用不同的算法求解0-1背包问题

阅读更多
一.动态规划求解0-1背包问题
/************************************************************************/
/* 0-1背包问题:
/*    给定n种物品和一个背包
/*        物品i的重量为wi,其价值为vi
/*        背包的容量为c
/*    应如何选择装入背包的物品,使得装入背包中的物品
/*    的总价值最大?
/*    注:在选择装入背包的物品时,对物品i只有两种选择,
/*        即装入或不装入背包。不能将物品i装入多次,也
/*        不能只装入部分的物品i。
/*
/* 1. 0-1背包问题的形式化描述:
/*    给定c>0, wi>0, vi>0, 0<=i<=n,要求找到一个n元的
/*    0-1向量(x1, x2, ..., xn), 使得:
/*            max sum_{i=1 to n} (vi*xi),且满足如下约束:
/*        (1) sum_{i=1 to n} (wi*xi) <= c
/*        (2) xi∈{0, 1}, 1<=i<=n
/*
/* 2. 0-1背包问题的求解
/*    0-1背包问题具有最优子结构性质和子问题重叠性质,适于
/*    采用动态规划方法求解
/*
/* 2.1 最优子结构性质
/*    设(y1,y2,...,yn)是给定0-1背包问题的一个最优解,则必有
/*    结论,(y2,y3,...,yn)是如下子问题的一个最优解:
/*            max sum_{i=2 to n} (vi*xi)
/*        (1) sum_{i=2 to n} (wi*xi) <= c - w1*y1
/*        (2) xi∈{0, 1}, 2<=i<=n
/*    因为如若不然,则该子问题存在一个最优解(z2,z3,...,zn),
/*    而(y2,y3,...,yn)不是其最优解。那么有:
/*        sum_{i=2 to n} (vi*zi) > sum_{i=2 to n} (vi*yi)
/*        且,w1*y1 + sum_{i=2 to n} (wi*zi) <= c
/*    进一步有:
/*        v1*y1 + sum_{i=2 to n} (vi*zi) > sum_{i=1 to n} (vi*yi)
/*        w1*y1 + sum_{i=2 to n} (wi*zi) <= c
/*    这说明:(y1,z2,z3,...zn)是所给0-1背包问题的更优解,那么
/*    说明(y1,y2,...,yn)不是问题的最优解,与前提矛盾,所以最优
/*    子结构性质成立。
/*
/* 2.2 子问题重叠性质
/*    设所给0-1背包问题的子问题 P(i,j)为:
/*            max sum_{k=i to n} (vk*xk)
/*        (1) sum_{k=i to n} (wk*xk) <= j
/*        (2) xk∈{0, 1}, i<=k<=n
/*    问题P(i,j)是背包容量为j、可选物品为i,i+1,...,n时的子问题
/*    设m(i,j)是子问题P(i,j)的最优值,即最大总价值。则根据最优
/*    子结构性质,可以建立m(i,j)的递归式:
/*        a. 递归初始 m(n,j)
/*        //背包容量为j、可选物品只有n,若背包容量j大于物品n的
/*        //重量,则直接装入;否则无法装入。
/*            m(n,j) = vn, j>=wn
/*            m(n,j) = 0, 0<=j<wn
/*        b. 递归式 m(i,j)
/*        //背包容量为j、可选物品为i,i+1,...,n
/*        //如果背包容量j<wi,则根本装不进物品i,所以有:
/*            m(i,j) = m(i+1,j), 0<=j<wi
/*        //如果j>=wi,则在不装物品i和装入物品i之间做出选择
/*            不装物品i的最优值:m(i+1,j)
/*            装入物品i的最优值:m(i+1, j-wi) + vi
/*            所以:
/*            m(i,j) = max {m(i+1,j), m(i+1, j-wi) + vi}, j>=wi
/*
/************************************************************************/



#define max(a,b) (((a) > (b)) ? (a) : (b))
#define min(a,b) (((a) < (b)) ? (a) : (b))
template <typename Type>
void Knapsack(Type* v, int *w, int c, int n, Type **m)
{
    //递归初始条件
    int jMax = min(w[n] - 1, c);
    for (int j=0; j<=jMax; j++) {
        m[n][j] = 0;
    }

    for (j=w[n]; j<=c; j++) {
        m[n][j] = v[n];
    }

    //i从2到n-1,分别对j>=wi和0<=j<wi即使m(i,j)
    for (int i=n-1; i>1; i--) {
        jMax = min(w[i] - 1, c);
        for (int j=0; j<=jMax; j++) {
            m[i][j] = m[i+1][j];
        }
        for (j=w[i]; j<=c; j++) {
            m[i][j] = max(m[i+1][j], m[i+1][j-w[i]]+v[i]);
        }
    }

    m[1][c] = m[2][c];
    if (c >= w[1]) {
        m[1][c] = max(m[1][c], m[2][c-w[1]]+v[1]);
    }

}

template <typename Type>
void TraceBack(Type **m, int *w, int c, int n, int* x)
{
    for (int i=1; i<n; i++) {
        if(m[i][c] == m[i+1][c]) x[i] = 0;
        else {
            x[i] = 1;
            c -= w[i];
        }
    }
    x[n] = (m[n][c])? 1:0;
}


int main(int argc, char* argv[])
{
    int n = 5;
    int w[6] = {-1, 2, 2, 6, 5, 4};
    int v[6] = {-1, 6, 3, 5, 4, 6};
    int c = 10;

    int **ppm = new int*[n+1];
    for (int i=0; i<n+1; i++) {
        ppm[i] = new int[c+1];
    }

    int x[6];

    Knapsack<int>(v, w, c, n, ppm);
    TraceBack<int>(ppm, w, c, n, x);

    return 0;
}
二.贪心算法求解0-1背包问题
1.贪心法的基本思路:
——从问题的某一个初始解出发逐步逼近给定的目标,以尽可能快的地求得更好的解。当达到某算法中的某一步不能再继续前进时,算法停止。
该算法存在问题:
1).不能保证求得的最后解是最佳的;
2).不能用来求最大或最小解问题;
3).只能求满足某些约束条件的可行解的范围。

实现该算法的过程:
从问题的某一初始解出发;
while 能朝给定总目标前进一步 do
   求出可行解的一个解元素;
由所有解元素组合成问题的一个可行解;

2.例题分析

1).[背包问题]有一个背包,背包容量是M=150。有7个物品,物品可以分割成任意大小。
要求尽可能让装入背包中的物品总价值最大,但不能超过总容量。

物品 A  B  C D E F G
重量 35  30  60  50  40  10  25 
价值  10  40  30 50  35  40  30

分析:
目标函数: ∑pi最大
约束条件是装入的物品总重量不超过背包容量:∑wi<=M( M=150)
(1)根据贪心的策略,每次挑选价值最大的物品装入背包,得到的结果是否最优?
(2)每次挑选所占空间最小的物品装入是否能得到最优解?
(3)每次选取单位容量价值最大的物品,成为解本题的策略。

<程序代码:>(环境:c++)
#include<iostream.h>
#define max 100 //最多物品数
void sort (int n,float a[max],float b[max]) //按价值密度排序
{
int j,h,k;
float t1,t2,t3,c[max];
for(k=1;k<=n;k++)
c[k]=a[k]/b[k];
for(h=1;h<n;h++)
for(j=1;j<=n-h;j++)
if(c[j]<c[j+1])
{t1=a[j];a[j]=a[j+1];a[j+1]=t1;
t2=b[j];b[j]=b[j+1];b[j+1]=t2;
t3=c[j];c[j]=c[j+1];c[j+1]=t3;
}
}
void knapsack(int n,float limitw,float v[max],float w[max],int x[max])
{float c1; //c1为背包剩余可装载重量
int i;
sort(n,v,w); //物品按价值密度排序
c1=limitw;
for(i=1;i<=n;i++)
{
if(w[i]>c1)break;
x[i]=1; //x[i]为1时,物品i在解中
c1=c1-w[i];
}
}
void main()
{int n,i,x[max];
float v[max],w[max],totalv=0,totalw=0,limitw;
cout<<"请输入n和limitw:";
cin>>n >>limitw;
for(i=1;i<=n;i++)
x[i]=0; //物品选择情况表初始化为0
cout<<"请依次输入物品的价值:"<<endl;
for(i=1;i<=n;i++)
cin>>v[i];
cout<<endl;
cout<<"请依次输入物品的重量:"<<endl;
for(i=1;i<=n;i++)
cin>>w[i];
cout<<endl;
knapsack (n,limitw,v,w,x);
cout<<"the selection is:";
for(i=1;i<=n;i++)
{
cout<<x[i];
if(x[i]==1)
totalw=totalw+w[i];
}
cout<<endl;
cout<<"背包的总重量为:"<<totalw<<endl; //背包所装载总重量
cout<<"背包的总价值为:"<<totalv<<endl; //背包的总价值
}
三.回溯算法求解0-1背包问题
1.0-l背包问题是子集选取问题。
    一般情况下,0-1背包问题是NP难题。0-1背包
问题的解空间可用子集树表示。解0-1背包问题的回溯法与装载问题的回溯法十分类
似。在搜索解空间树时,只要其左儿子结点是一个可行结点,搜索就进入其左子树。当
右子树有可能包含最优解时才进入右子树搜索。否则将右子树剪去。设r是当前剩余
物品价值总和;cp是当前价值;bestp是当前最优价值。当cp+r≤bestp时,可剪去右
子树。计算右子树中解的上界的更好方法是将剩余物品依其单位重量价值排序,然后
依次装入物品,直至装不下时,再装入该物品的一部分而装满背包。由此得到的价值是
右子树中解的上界。
2.解决办法思路:
    为了便于计算上界,可先将物品依其单位重量价值从大到小排序,此后只要顺序考
察各物品即可。在实现时,由bound计算当前结点处的上界。在搜索解空间树时,只要其左儿子节点是一个可行结点,搜索就进入左子树,在右子树中有可能包含最优解是才进入右子树搜索。否则将右子树剪去。

    回溯法是一个既带有系统性又带有跳跃性的的搜索算法。它在包含问题的所有解的解空间树中,按照深度优先的策略,从根结点出发搜索解空间树。算法搜索至解空间树的任一结点时,总是先判断该结点是否肯定不包含问题的解。如果肯定不包含,则跳过对以该结点为根的子树的系统搜索,逐层向其祖先结点回溯。否则,进入该子树,继续按深度优先的策略进行搜索。回溯法在用来求问题的所有解时,要回溯到根,且根结点的所有子树都已被搜索遍才结束。而回溯法在用来求问题的任一解时,只要搜索到问题的一个解就可以结束。这种以深度优先的方式系统地搜索问题的解的算法称为回溯法,它适用于解一些组合数较大的问题。
2.算法框架:
    a.问题的解空间:应用回溯法解问题时,首先应明确定义问题的解空间。问题的解空间应到少包含问题的一个(最优)解。
    b.回溯法的基本思想:确定了解空间的组织结构后,回溯法就从开始结点(根结点)出发,以深度优先的方式搜索整个解空间。这个开始结点就成为一个活结点,同时也成为当前的扩展结点。在当前的扩展结点处,搜索向纵深方向移至一个新结点。这个新结点就成为一个新的活结点,并成为当前扩展结点。如果在当前的扩展结点处不能再向纵深方向移动,则当前扩展结点就成为死结点。换句话说,这个结点不再是一个活结点。此时,应往回移动(回溯)至最近的一个活结点处,并使这个活结点成为当前的扩展结点。回溯法即以这种工作方式递归地在解空间中搜索,直至找到所要求的解或解空间中已没有活结点时为止。
3.运用回溯法解题通常包含以下三个步骤:
    a.针对所给问题,定义问题的解空间;
    b.确定易于搜索的解空间结构;
    c.以深度优先的方式搜索解空间,并且在搜索过程中用剪枝函数避免无效搜索;
#include<iostream>

using namespace std;


class Knap
{
friend int Knapsack(int p[],int w[],int c,int n );

public:
void print()
{
   
for(int m=1;m<=n;m++)
   {
    cout<<bestx[m]<<" ";
   }
   cout<<endl;
};

private:
int Bound(int i);
void Backtrack(int i);

int c;//背包容量
int n; //物品数
int *w;//物品重量数组
int *p;//物品价值数组
int cw;//当前重量
int cp;//当前价值
int bestp;//当前最优值
int *bestx;//当前最优解
int *x;//当前解

};


int Knap::Bound(int i)
{
//计算上界
int cleft=c-cw;//剩余容量
int b=cp;
//以物品单位重量价值递减序装入物品
while(i<=n&&w[i]<=cleft)
{
   cleft-=w[i];
   b+=p[i];
   i++;
}
//装满背包
if(i<=n)
   b+=p[i]/w[i]*cleft;
return b;
}


void Knap::Backtrack(int i)
{
if(i>n)
{
    if(bestp<cp)
{    
    for(int j=1;j<=n;j++)
     bestx[j]=x[j];
     bestp=cp;
}
return;
}
if(cw+w[i]<=c) //搜索左子树
{            
      x[i]=1;
   cw+=w[i];
   cp+=p[i];
   Backtrack(i+1);
   cw-=w[i];
   cp-=p[i];
}
   if(Bound(i+1)>bestp)//搜索右子树
   {
       x[i]=0;
    Backtrack(i+1);
   }

}


class Object
{
friend int Knapsack(int p[],int w[],int c,int n);
public:
int operator<=(Object a)const
{
return (d>=a.d);
}

private:
int ID;
float d;
};


int Knapsack(int p[],int w[],int c,int n)
{
//为Knap::Backtrack初始化
int W=0;
int P=0;
int i=1;
Object *Q=new Object[n];
for(i=1;i<=n;i++)
{
Q[i-1].ID=i;
Q[i-1].d=1.0*p[i]/w[i];
P+=p[i];
W+=w[i];
}
if(W<=c)
   return P;//装入所有物品
//依物品单位重量排序
float f;
for( i=0;i<n;i++)
for(int j=i;j<n;j++)
{
   if(Q[i].d<Q[j].d)
   {
     f=Q[i].d;
   Q[i].d=Q[j].d;
   Q[j].d=f;
   }


}

Knap K;
K.p = new int[n+1];
    K.w = new int[n+1];
K.x = new int[n+1];
K.bestx = new int[n+1];
K.x[0]=0;
K.bestx[0]=0;
for( i=1;i<=n;i++)
{
K.p[i]=p[Q[i-1].ID];
K.w[i]=w[Q[i-1].ID];
}
K.cp=0;
K.cw=0;
K.c=c;
K.n=n;
K.bestp=0;
//回溯搜索
K.Backtrack(1);
    K.print();
    delete [] Q;
delete [] K.w;
delete [] K.p;
return K.bestp;

}

void main()
{
int *p;
int *w;
    int c=0;
int n=0;
int i=0;
char k;
cout<<"0-1背包问题——回溯法 "<<endl;
cout<<"     by zbqplayer    "<<endl;
while(k)
{
cout<<"请输入背包容量(c):"<<endl;
cin>>c;
cout<<"请输入物品的个数(n):"<<endl;
    cin>>n;
p=new int[n+1];
w=new int[n+1];
p[0]=0;
w[0]=0;

cout<<"请输入物品的价值(p):"<<endl;
for(i=1;i<=n;i++)
   cin>>p[i];

cout<<"请输入物品的重量(w):"<<endl;
for(i=1;i<=n;i++)
   cin>>w[i];

cout<<"最优解为(bestx):"<<endl;
cout<<"最优值为(bestp):"<<endl;
cout<<Knapsack(p,w,c,n)<<endl;
    cout<<"[s] 重新开始"<<endl;
cout<<"[q] 退出"<<endl;
cin>>k;
}
四.分支限界法求解0-1背包问题
1.问题描述:已知有N个物品和一个可以容纳M重量的背包,每种物品I的重量为WEIGHT,一个只能全放入或者不放入,求解如何放入物品,可以使背包里的物品的总效益最大。

2.设计思想与分析:对物品的选取与否构成一棵解树,左子树表示不装入,右表示装入,通过检索问题的解树得出最优解,并用结点上界杀死不符合要求的结点。

#include <iostream.h>

struct good
{
int weight;
int benefit;
int flag;//是否可以装入标记
};

int number=0;//物品数量
int upbound=0;
int curp=0, curw=0;//当前效益值与重量
int maxweight=0;
good *bag=NULL;

void Init_good()
{
bag=new good [number];

for(int i=0; i<number; i++)
{
  cout<<"请输入第件"<<i+1<<"物品的重量:";
  cin>>bag[i].weight;
  cout<<"请输入第件"<<i+1<<"物品的效益:";
  cin>>bag[i].benefit;
  bag[i].flag=0;//初始标志为不装入背包
  cout<<endl;
}

}

int getbound(int num, int *bound_u)//返回本结点的c限界和u限界
{
for(int w=curw, p=curp; num<number && (w+bag[num].weight)<=maxweight; num++)
{
  w=w+bag[num].weight;
  p=w+bag[num].benefit;
}

*bound_u=p+bag[num].benefit;
return ( p+bag[num].benefit*((maxweight-w)/bag[num].weight) );
}

void LCbag()
{
int bound_u=0, bound_c=0;//当前结点的c限界和u限界

for(int i=0; i<number; i++)//逐层遍历解树决定是否装入各个物品
{
  if( ( bound_c=getbound(i+1, &bound_u) )>upbound )//遍历左子树
    upbound=bound_u;//更改已有u限界,不更改标志 

  if( getbound(i, &bound_u)>bound_c )//遍历右子树
  //若装入,判断右子树的c限界是否大于左子树根的c限界,是则装入
  {
   upbound=bound_u;//更改已有u限界
   curp=curp+bag[i].benefit;
   curw=curw+bag[i].weight;//从已有重量和效益加上新物品
   bag[i].flag=1;//标记为装入
  }
}

}

void Display()
{

cout<<"可以放入背包的物品的编号为:";
for(int i=0; i<number; i++)
  if(bag[i].flag>0)
   cout<<i+1<<" ";
  cout<<endl;
  delete []bag;
}
分享到:
评论
1 楼 被判孤寂 2009-12-08  
很经典的算法.

相关推荐

    遗传算法求解0-1背包问题matlab代码.zip

    遗传算法求解0-1背包问题matlab代码

    一种改进的模拟退火算法求解0-1背包问题

    ### 一种改进的模拟退火算法求解0-1背包问题 #### 背景与问题定义 0-1背包问题是一种经典的NP完全问题,在工程、经济、资源分配等领域广泛应用。该问题的核心在于如何在有限的背包容量下,从一组物品中选择若干...

    遗传算法求解0-1背包模型的MATLAB代码

    在MATLAB环境中实现遗传算法求解0-1背包问题,主要涉及以下几个方面: 1. **初始化种群**:随机生成一定数量的个体,每个个体代表一种可能的物品选择方案,用二进制向量表示。例如,如果物品总数为n,那么个体可以...

    基于遗传算法的求解0-1背包问题

    基于遗传算法的求解0-1背包问题 遗传算法是一种生物启发式优化算法,通过模拟自然选择和进化机制来搜索最优解。遗传算法在解决复杂优化问题方面具有广泛的应用前景,背包问题作为一个典型的NP难问题,利用遗传算法...

    动态规划求解0-1背包问题的改进算法完整解释

    动态规划求解0-1背包问题的改进算法完整解释 在计算机算法设计与分析中,动态规划法是解决背包问题的常用方法之一。所谓背包问题,是指在有限的背包容量下,如何选择物品来达到最大价值的问题。在本文中,我们将对...

    带权重的贪心萤火虫算法求解0-1背包问题

    参考文献:任静敏,潘大志《带权重的贪心萤火虫算法求解0-1背包问题》,用MATLAB实现改进萤火虫算法(WGFA),对基本的萤火虫算法进行改进,加入线性递减惯性权重,用贪心算法修复不可行解,加入变异算子提高全局...

    分支定界算法求解0-1背包问题(附MATLAB代码).rar

    总的来说,分支定界算法是解决0-1背包问题的有效方法,结合MATLAB代码,学习者不仅可以理解算法原理,还能动手实践,加深对问题解决过程的理解。通过掌握0-1背包问题及其求解策略,有助于提升在组合优化领域的理论...

    遗传算法求解0-1背包问题

    在0-1背包问题的遗传算法求解过程中,个体通常表示为一个二进制序列,每个位对应一个物品,1表示选取该物品,0则表示不选。种群中的每个个体代表一个可能的解,即一组物品的选择。 遗传算法的基本步骤包括: 1. ...

    动态规划法求解0-1背包问题实验报告.pdf

    总结来说,动态规划法求解0-1背包问题的关键在于构建正确的状态转移方程,并通过填表的方式逐步计算出所有子问题的最大价值。这种思想不仅可以应用于背包问题,还可以广泛应用于其他优化问题,如最长公共子序列、...

    使用遗传算法 在Python中解决 0-1 背包问题的简单方法_python_代码_下载

    总的来说,遗传算法在0-1背包问题中的应用展示了其在复杂优化问题中的强大能力。通过模拟自然选择过程,它能够在相对较少的计算时间内找到接近最优解的解决方案。同时,Python作为一门广泛使用的编程语言,为实现和...

    遗传算法求解0-1背包问题matlab代码 这是遗传算法用来求解0-1背包

    在本案例中,提供的MATLAB代码实现了遗传算法求解0-1背包问题的过程。用户可以从中学习如何定义适应度函数、编码方案、选择、交叉和变异策略。通过运行代码,可以观察算法在不同参数设置下的性能,以及解的质量。...

    模拟退火算法解决0-1背包问题

    利用MATLAB退火算法解决0-1背包问题。数据直接在主函数内,如有需要,直接替换即可

    贪心算法解决0-1背包问题

    然而,贪心算法并不总是能保证找到0-1背包问题的全局最优解。例如,当存在两件物品A和B,A的价值密度高于B,但将A和B一起放入背包可以得到更高的总价值,这时贪心算法会错过这个更好的组合。因此,对于某些特定的...

    利用回溯法解0-1背包问题讲解

    * 算法描述:0-1背包问题是子集选取问题。一般情况下,0-1背包问题是NP难的。0-1背包问题的解空间可用子集树表示。 * 在搜索解空间树时,只要其左儿子结点是一个可行结点,搜索就进入左子树。当右子树中有可能包含最...

    matlab遗传算法求解0-1背包问题.zip

    0-1背包问题是一种经典的组合优化...通过这个MATLAB实现,我们可以直观地理解遗传算法如何应用于解决0-1背包问题,并能灵活调整算法参数以优化性能。这种方法不仅适用于背包问题,还可以推广到其他类似的组合优化问题。

    动态规划法和回溯法求0-1背包问题

    回溯法求解0-1背包问题 **基本思想** 回溯法是一种基于试探性搜索的算法,它通过不断地探索和回退来寻找问题的解。在解决0-1背包问题时,回溯法采用的是深度优先搜索策略,同时利用限界函数来剪枝,避免无效的...

    求解0-1背包问题的快速混合遗传算法

    ### 求解0-1背包问题的快速混合遗传算法 #### 一、背包问题的数学描述 0/1背包问题(Knapsack Problem, KP)是一种典型的NP完全问题,在计算机科学与运筹学领域中占有重要地位。该问题描述了一个背包能够承受的...

    遗传算法回溯算法解0--1背包问题

    调用`zhou`函数求解0-1背包问题后,打印出最大价值以及哪些物品被选中放入背包。 #### 五、结论 回溯算法通过其灵活的搜索策略,能够在复杂的组合优化问题中有效地找到最优解,尤其是在0-1背包问题这种约束条件...

    遗传算法解决 0-1背包问题

    在解决0-1背包问题时,遗传算法可以将每个物品表示为一个个体,个体的基因由0和1组成,1表示选取该物品,0表示不选。通过对种群进行迭代,逐步优化解的质量。 在DELPHI编程环境中实现遗传算法,首先需要定义一个...

Global site tag (gtag.js) - Google Analytics