回溯法之01背包问题

回溯法之01背包问题一、问题n皇后问题的解空间树是一颗排列树,而01背包问题的解空间树应该是一颗子集树。再简述下该问题:有n件物品和一个容量为c的背包。第i件物品的价值是v[i],重量是w[i]。求解将哪些物品装入背包可使价值总和最大。所谓01背包,表示每一个物品看成一个整体,要么全部装入,要么都不装入。这里n=5,c=10,w={2,2,6,5,4},v={6,3,5,4,6}。01背…

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE稳定放心使用

一、问题

n皇后问题的解空间树是一颗排列树,而01背包问题的解空间树应该是一颗子集树。再简述下该问题:有n件物品和一个容量为c的背包。第i件物品的价值是v[i],重量是w[i]。求解将哪些物品装入背包可使价值总和最大。所谓01背包,表示每一个物品看成一个整体,要么全部装入,要么都不装入。这里n=5, c=10, w={2, 2, 6, 5, 4}, v={6, 3, 5, 4, 6}。

01背包属于找最优解问题,用回溯法需要构造解的子集树。在搜索状态空间树时,只要左子结点是可一个可行结点,搜索就进入其左子树。对于右子树时,先计算上界函数,以判断是否将其剪枝。为了更好地计算和运用上界函数剪枝,可以先将物品按照其单位重量价值从大到小排序,此后就按照顺序考虑各个物品。

剪枝的两种情况:

①左结点深度搜索的条件是当前物品装得下,即cw+w[i]<=c

②将当前剩余所有物品都装入背包,获得的总价值也没能大于当前已经求得的最大价值bestp时

二、c++代码

#include <iostream>
#include <cmath>
using namespace std;


int n;//物品数量
double c;//背包容量
double v[100];//各个物品的价值
double w[100];//各个物品的重量
double cw = 0.0;//当前背包重量
double cp = 0.0;//当前背包中物品价值
double bestp = 0.0;//当前最优价值
double perp[100];//物品按单位重量价值排序后
int x[100];//是否装入,为0或1 
int best[100];//记录最优解,为0或1 
double v2[100];//临时存放各个物品的价值
double w2[100];//临时存放各个物品的重量


//交换两元素 
void swap(int i,int j,double arr[]){
     double t;
     t=arr[i];arr[i]=arr[j];arr[j]=t;                
     }
 
//快速排序算法 
void quicksort(int p,int q,double arr[],double key){
    int i,j;
    i=p;
    j=q;
    if(p>=q){
         return;
         }
    
    while(1){
            while(j>=p && arr[j]<key){j--;}
            if(j<=i) 
            break;
            swap(i,j,arr);
            swap(i,j,w2);
            swap(i,j,v2);

            
            while(i<=q && arr[i]>=key){i++;}                                  
            if(j<=i) 
            break;
            swap(i,j,arr);
            swap(i,j,w2);
            swap(i,j,v2);
          
            
            }
            
     quicksort(p,j-1,arr,arr[p]);
     quicksort(j+1,q,arr,arr[j+1]);   
    
    }

//按单位价值排序
void knapsack(int t)
{       
   quicksort(t,n,perp,perp[t]);    
}

//回溯函数
//每个结点的左右子树都要判断,因为装或不装两种情况都要考虑 
void backtrack(int i)
{
    double bound(int i);
   
    if(i>n)//到达叶结点,得出一个解 
    {
    bestp = cp;
    for(int k=1;k<=n;k++)
    {      
       best[k]=x[k];//把最优解记录下来      
    }
       return;
    }
    
    if(cw+w[i]<=c)//搜索左子树 
    {
       cw+=w[i];
       cp+=v[i];
       x[i]=1;
       backtrack(i+1);
       cw-=w[i];
       cp-=v[i];
       x[i]=0;
    }   
    
   if(bound(i+1)>bestp)//搜索右子树,必要时剪枝 
       backtrack(i+1);
}

//计算上界函数
double bound(int i)
{
    double leftw= c-cw;
    double b =cp;
    
    for(int k=i;k<=n;k++){//剩余物品重量、价值分别存在w2、v2数组中 
             w2[k]=w[k];
             v2[k]=v[k];
             } 
    knapsack(i);//将剩余物品按单位重量价值排序
    
    while(i<=n&&w2[i]<=leftw)//将剩余已排好序的物品装入背包 
    {
       leftw-=w2[i];
       b+=v2[i];
       i++;
    }
    if(i<=n)
       b+=v2[i]/w2[i]*leftw;
    return b;
}


int main()
{
    int i; 
    cin >> n >> c;//输入物品数量n、背包容量c   
    for(i=1;i<=n;i++)
    {
       cin >> w[i] >> v[i];//输入各个物品重量wi、价值vi  
       perp[i]=v[i]/w[i];  
       w2[i]=v2[i]=0;       
       x[i]=0;
       best[i]=0;
    }
        
    backtrack(1);
    cout << "最大价值为:" << bestp << endl;
    cout << "需要装入的物品编号是:" << endl;
    for(i=1;i<=n;i++)
    {
       if(best[i])
           cout << i << " ";
    }
    cout << endl;
    system("pause"); 
    return 0;
}

运行结果:

回溯法之01背包问题

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/180962.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • mysql之视图、索引

    mysql之视图、索引视图 什么是视图 视图(View)是一种虚拟存在的表,同真实表一样,视图也由列和行构成,但视图并不实际存在于数据库中。行和列的数据来自于定义视图的查询中所使用的表,并且还是在使用视图时动态生成的。数据库中只存放了视图的定义,并没有存放视图中的数据,这些数据都存放在定义视图查询所引用的真实表中。使用视图查询数据时,数据库会从真实表中取出对应的数据。因此,视图中的数据是依赖于真实表中的数据的。一旦真实表中的数据发生改变,显示在视图中的数据也会发生改变。 视图的作用 定制用户数据,聚焦

    2022年7月22日
    13
  • vue 使用vue-i18n实现中英文语言切换,以及动态添加中英文「建议收藏」

    vue 使用vue-i18n实现中英文语言切换,以及动态添加中英文「建议收藏」一、安装。npminstallvue-i18n二、使用。(这里只写了核心代码)引入://引入插件和语言包importVueI18nfrom’vue-i18n’importzhfrom’@/i18n/langs/zh’importenfrom’@/i18n/langs/en’Vue.use(VueI18n);zh文件:exportdefault{placeStopOrder:”下止损单”,}en文件:exportdef

    2022年5月8日
    146
  • iframe标签(页面嵌套)

    开发工具与关键技术:VS<iframe>作者:听民谣的老猫撰写时间:2019/6/1018:15上面两张图是两个不同的页面但是它们的基本框架都是一样,每点击一次左边的导航栏改变的都是中间的内容区域。也就是说共同的框架都是没有改变的,改变的是中间的内容。有没有什么方法可以不改变外面的基本框架只改变中间的内容???我们可以用页面嵌套方法来达到这一要求。页面嵌…

    2022年4月8日
    82
  • 通过全备+主从同步恢复被drop的库或表

    通过全备+主从同步恢复被drop的库或表

    2021年6月9日
    88
  • mysql临时表更新_MySql 临时表

    mysql临时表更新_MySql 临时表今天在项目中遇到一个 当 mysql 的 in 语句中数据量很大时 建立一个临时表的例子 于是楼主整理了一下关于临时表的知识 与大家分享一下 首先 临时表只在当前连接可见 当关闭连接时 Mysql 会自动删除表并释放所有空间 因此在不同的连接中可以创建同名的临时表 并且操作属于本连接的临时表 建立临时表 CREATETEMPOR table idVARCHAR 32 N

    2025年10月6日
    4
  • 电商后台管理系统项目介绍和项目初始化

    电商后台管理系统项目介绍和项目初始化目录一电商项目基本业务概述二电商后台管理系统功能三电商后台管理系统的开发模式(前后端分离)四电商后台管理系统的技术选型五前端项目初始化步骤六创建新项目七安装element-UI插件八配置element-UI为按需导入九安装axios十码云上创建vue_shop仓库,将初始化项目托管到码云一电商项目基本业务概述根据不同的应用场景,电商系统一般都提供了PC端、移动APP、移动Web、微信小程序等多种终端访问方式。二电商后…

    2022年6月4日
    45

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号