分支限界法之LC 0/1背包

本文介绍了一个经典的背包问题求解算法,通过构建解树并利用限界条件进行剪枝,找到在给定容量限制下使总效益最大的物品组合。
  1. 1.问题描述:已知有N个物品和一个可以容纳M重量的背包,每种物品I的重量为WEIGHT,一个只能全放入或者不放入,求解如何放入物品,可以使背包里的物品的总效益最大。
  2. 2.设计思想与分析:对物品的选取与否构成一棵解树,左子树表示不装入,右表示装入,通过检索问题的解树得出最优解,并用结点上界杀死不符合要求的结点。
  3. (多谢shadow同学提供该算法)
  4. #include <iostream.h>
  5. struct good
  6. {
  7. int weight;
  8. int benefit;
  9. int flag;//是否可以装入标记
  10. };
  11. int number=0;//物品数量
  12. int upbound=0;
  13. int curp=0, curw=0;//当前效益值与重量
  14. int maxweight=0;
  15. good *bag=NULL;
  16. void Init_good()
  17. {
  18. bag=new good [number];
  19. for(int i=0; i<number; i++)
  20. {
  21. cout<<"请输入第件"<<i+1<<"物品的重量:";
  22. cin>>bag[i].weight;
  23. cout<<"请输入第件"<<i+1<<"物品的效益:";
  24. cin>>bag[i].benefit;
  25. bag[i].flag=0;//初始标志为不装入背包
  26. cout<<endl;
  27. }
  28. }
  29. int getbound(int num, int *bound_u)//返回本结点的c限界和u限界
  30. {
  31. for(int w=curw, p=curp; num<number && (w+bag[num].weight)<=maxweight; num++)
  32. {
  33. w=w+bag[num].weight;
  34. p=w+bag[num].benefit;
  35. }
  36. *bound_u=p+bag[num].benefit;
  37. return ( p+bag[num].benefit*((maxweight-w)/bag[num].weight) );
  38. }
  39. void LCbag()
  40. {
  41. int bound_u=0, bound_c=0;//当前结点的c限界和u限界
  42. for(int i=0; i<number; i++)//逐层遍历解树决定是否装入各个物品
  43. {
  44. if( ( bound_c=getbound(i+1, &bound_u) )>upbound )//遍历左子树
  45. upbound=bound_u;//更改已有u限界,不更改标志
  46. if( getbound(i, &bound_u)>bound_c )//遍历右子树
  47. //若装入,判断右子树的c限界是否大于左子树根的c限界,是则装入
  48. {
  49. upbound=bound_u;//更改已有u限界
  50. curp=curp+bag[i].benefit;
  51. curw=curw+bag[i].weight;//从已有重量和效益加上新物品
  52. bag[i].flag=1;//标记为装入
  53. }
  54. }
  55. }
  56. void Display()
  57. {
  58. cout<<"可以放入背包的物品的编号为:";
  59. for(int i=0; i<number; i++)
  60. if(bag[i].flag>0)
  61. cout<<i+1<<" ";
  62. cout<<endl;
  63. delete []bag;
  64. }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值