蓝桥杯每日一题------背包问题(三)

前言

之前求的是在特点情况下选择一些物品让其价值最大,这里求的是方案数以及具体的方案。

背包问题求方案数

在这里插入图片描述
既然要求方案数,那么就需要一个新的数组来记录方案数。动态规划步骤如下,
定义dp数组
第一步:缩小规模。考虑n个物品,那我就先考虑1个物品,在考虑2个物品…,需要一个维度表示当前考虑的物品个数。
第二步:限制。所选物品个数不能超过物品容量,那么需要一个维度记录当前背包的容量。
第三步:写出dp数组。f[i][j]表示当前考虑了前i个物品,背包容量为j价值最大时的方案数。
第四步:推状态转移方程。f[i][j]应该从哪里转移过来呢,必然是从前i-1个物品转移,我要考虑两种情况,对于第i个物品,可以选择要它,也可以不要它。
(1)如果要第i个物品,f[i][j]=f[i-1][j-v[i]]。
(2)如果不要第i个物品,f[i][j]=f[i-1][j]。
(3)如果要和不要都可以获得最大价值,f[i][j]=f[i-1][j-v[i]]+f[i-1][j]。
那么在原先的dp数组进行转移的时候,我们要记录他到底是从哪个状态转移过来的,不能只单纯取一个最大值。代码如下。

int l = dp[j];int r = dp[j-v[i]] + w[i];dp[j] = Math.max(l, r);if(l > r) {f[j] = f[j];}else if (l < r) {f[j] = f[j-v[i]];}else{f[j] = f[j] + f[j-v[i]];}

同时要注意f数组的初始化,当考虑前0个物品时,方案是啥也不选,所以方案数是1,初始化代码如下,

for(int i = 0;i < k+1;i++){f[i] = 1;}

完整代码如下,

import java.util.Scanner;
public class Main {
public static void main(String[] args) {Scanner scanner = new Scanner(System.in);int n = scanner.nextInt();int k = scanner.nextInt();int v[] = new int[n+1];int w[] = new int[n+1];int dp[] = new int[k+1];int f[] = new int[k+1];for (int i = 1; i < n+1; i++) {v[i] = scanner.nextInt();w[i] = scanner.nextInt();}for(int i = 0;i < k+1;i++) f[i] = 1;int mod = 1000000007;for (int i = 1; i < n+1; i++) {for (int j = k; j >= v[i]; j--) {int l = dp[j];int r = dp[j-v[i]] + w[i];dp[j] = Math.max(l, r);if(l > r) {f[j] = f[j];}else if (l < r) {f[j] = f[j-v[i]];}else{f[j] = f[j] + f[j-v[i]];}f[j] %= mod;}}System.out.println(f[k]);
}
}

背包问题求具体方案

在这里插入图片描述
正常求dp数组,求完后逆序推回去就行,对于dp[n][m],如果dp[n][m]=dp[n][m-v[i]]+w[i],那么第i个物品被选择,然后再接着往后求dp[n][m-v[i]]。
那么如何保证字典序最小?回顾一下求dp数组的过程,我们是优先考虑的字典序小的物品,但是最后求的时候我们是倒序推导,这样反而字典序大的物品会优先被输出,所以更改一下字典序小的物品放在后面求。
全部代码如下,

import java.util.Scanner;
public class Main {
public static void main(String[] args) {Scanner scanner = new Scanner(System.in);int n = scanner.nextInt();int V = scanner.nextInt();int[] v = new int[n+1];int[] w = new int[n+1];for (int i = 1; i < w.length; i++) {v[i] = scanner.nextInt();w[i] = scanner.nextInt();}int[][] dp = new int[n+2][V+1];for (int i = n; i > 0; i--) {for (int j = 0; j < V+1; j++) {dp[i][j] = dp[i+1][j];if(v[i] <= j) {dp[i][j] = Math.max(dp[i][j], dp[i+1][j-v[i]] + w[i]);}}}int vv = V;for (int i = 1; i < n+1&&vv>0; i++) {if(vv >= v[i] && dp[i][vv] == dp[i+1][vv-v[i]] + w[i]) {System.out.print(i + " ");vv -= v[i];}}
}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/465063.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

锐捷(十九)锐捷设备的接入安全

1、PC1的IP地址和mac地址做全局静态ARP绑定; 全局下&#xff1a;address-bind 192.168.1.1 mac&#xff08;pc1&#xff09; G0/2:ip verify source port-securityarp-check 2、PC2的IP地址和MAC地址做全局IPMAC绑定&#xff1a; Address-bind 192.168.1.2 0050.7966.6807Ad…

144. 二叉树的前序遍历

给你二叉树的根节点 root &#xff0c;返回它节点值的 前序 遍历。 示例 1&#xff1a; 输入&#xff1a;root [1,null,2,3] 输出&#xff1a;[1,2,3]示例 2&#xff1a; 输入&#xff1a;root [] 输出&#xff1a;[]示例 3&#xff1a; 输入&#xff1a;root [1] 输出&a…

每日一个shell脚本之自动化采集监控指标+登录欢迎

每日一个shell脚本之自动化采集监控指标登录欢迎 效果图参上 源码奉上 #!/usr/bin/bashclear#空闲内存Frfree -h | awk NR2{print $4}#已用内存Usfree -h | awk NR2{print $3}#系统存储空间Us_systemdf -Th | grep /dev/ | tail -1 | awk {print $4}Us_freedf -Th | grep /de…

【Linux进程间通信】用管道实现简单的进程池、命名管道

【Linux进程间通信】用管道实现简单的进程池、命名管道 目录 【Linux进程间通信】用管道实现简单的进程池、命名管道为什么要实现进程池&#xff1f;代码实现命名管道创建一个命名管道 理解命名管道匿名管道与命名管道的区别命名管道的打开规则 作者&#xff1a;爱写代码的刚子…

vue+springboot前后端视频文件等的上传与展示(基于七牛云)

前言&#xff1a;在初步说明完成功能之前&#xff0c;我会把重要的部分说明下。后续我会细化。 vue视频文件上传 其实这里和图片这些文件就是一样的。因为上传只是把我们想在云端展示的文件按等传输到云端的bucket。然后方便网站去请求引用。 有人问我我就说明下。这种东西无…

从Socket中解析Http协议实现通信

在网络协议中&#xff0c;Socket是连接应用层和运输层的中间层&#xff0c;主要作用为了通信。Http协议是应用层上的封装协议。我们可以通过Http协议的规范解析Socket中数据&#xff0c;完成Http通信。 首先&#xff0c;我们先回顾一下Http协议的规范。主要复习一下&#xff0c…

前端工程化面试题 | 01.精选前端工程化高频面试题

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…

JAVA设计模式之代理模式详解

代理模式 1 代理模式介绍 在软件开发中,由于一些原因,客户端不想或不能直接访问一个对象,此时可以通过一个称为"代理"的第三者来实现间接访问.该方案对应的设计模式被称为代理模式. 代理模式(Proxy Design Pattern ) 原始定义是&#xff1a;让你能够提供对象的替代…

Java String源码剖析+面试题整理

由于字符串操作是计算机程序中最常见的操作之一&#xff0c;在面试中也是经常出现。本文从基本用法出发逐步深入剖析String的结构和性质&#xff0c;并结合面试题来帮助理解。 String基本用法 在Java中String的创建可以直接像基本类型一样定义&#xff0c;也可以new一个 Str…

配备Apple T2 安全芯片的 Mac 机型及T2芯片mac电脑U盘装系统教程

T2 芯片为 Mac 提供了一系列功能&#xff0c;例如加密储存和安全启动功能、增强的图像信号处理功能&#xff0c;以及适用于触控 ID 数据的安全保护功能。哪些电脑配备了 T2 安全芯片呢&#xff0c;T2芯片mac电脑又如何重装系统呢&#xff1f;跟随小编一起来看看吧&#xff01; …

2024-2-11-复习作业

1> 要求&#xff1a; 源代码&#xff1a; #include <stdio.h> int fun(int n) {if(n0) return 1;return n*fun(n-1); } int main(int argc, char const *argv[]) {/* code */int n;printf("enter n :");scanf("%d",&n);int sfun(n);printf(…

[word] word分割线在哪里设置 #其他#经验分享

word分割线在哪里设置 在工作中有些技巧&#xff0c;可以快速提高工作效率&#xff0c;解决大部分工作&#xff0c;今天给大家分享word分割线在哪里设置的小技能&#xff0c;希望可以帮助到你。 1、快速输入分割线 输入三个【_】按下回车就是一条长直线&#xff0c;同样分别…