news 2026/2/26 10:34:14

(新卷,200分)- MELON的难题(Java JS Python C)

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
(新卷,200分)- MELON的难题(Java JS Python C)

(新卷,200分)- MELON的难题(Java & JS & Python & C)

题目描述

MELON有一堆精美的雨花石(数量为n,重量各异),准备送给S和W。MELON希望送给俩人的雨花石重量一致,请你设计一个程序,帮MELON确认是否能将雨花石平均分配。

输入描述

第1行输入为雨花石个数:n, 0 < n < 31。

第2行输入为空格分割的各雨花石重量:m[0] m[1] ….. m[n - 1], 0 < m[k] < 1001。

不需要考虑异常输入的情况。

输出描述

如果可以均分,从当前雨花石中最少拿出几块,可以使两堆的重量相等;如果不能均分,则输出-1。

用例
输入4
1 1 2 2
输出2
说明

输入第一行代表共4颗雨花石,

第二行代表4颗雨花石重量分别为1、1、2、2。

均分时只能分别为1,2,需要拿出重量为1和2的两块雨花石,所以输出2。

输入10
1 1 1 1 1 9 8 3 7 10
输出3
说明

输入第一行代表共10颗雨花石,

第二行代表4颗雨花石重量分别为1、1、1、1、1、9、8、3、7、10 。

均分时可以1,1,1,1,1,9,7和10,8,3,也可以1,1,1,1,9,8和10,7,3,1,或者其他均分方式,但第一种只需要拿出重量为10,8,3的3块雨花石,第二种需要拿出4块,所以输出3(块数最少)。

题目解析

我的解题思路如下:

首先,将所有雨花石重量之和求出,设为sum,

  • 如果sum % 2 != 0,则说明无法平分,直接返回-1。
  • 如果sum % 2 == 0,则说明“可能”平分。

如果“可能”平分,此时,我们可以将本问题转化为:01背包问题中“装满背包的最少物品数问题”。

其中:

  • 背包承重 = sum / 2
  • 物品 = 所有雨花石
  • 物品重量 = 雨花石重量

如果大家学习过01背包,其实状态转移方程非常容易推导:

我们假设dp[i][j] 表示从 0 ~ i 物品中选择,能装满背包承重 j 的最少物品数量。那么:

  • 如果第 i 个物品(重量为w)选择的话,则 dp[i][j] = dp[i-1][j - w] + 1;(ps:+1代表背包装入了第i个物品)
  • 如果第 i 个物品不选的话,则 dp[i][j] = dp[i-1][j];

最终 dp[i][j] 取最小值即可,即:dp[i][j] = min(dp[i-1][j],dp[i-1][j - w] + 1)

另外,我们需要注意dp的初始化,因为后面要求最少物品数量,因此如果将dp[0][0] ~ dp[0][bag] 初始化为0,则会影响后续取最小值(ps:0必然是最少的数量)。

因为我们应该将dp[0][0] ~ dp[0][bag]初始化为一个不可能的较大值,这里由于背包承重是sum/2,因此背包绝对不可能装入 n 个雨花石,因为n个雨花石的重量之和为sum。


根据考友指正,上面逻辑中对于dp[0][0] ~ dp[0][bag]的初始化存在问题,比如下面用例

3 3 1 2

按照上面逻辑dp[0][0] ~ dp[0][bag]全部会被初始化为3,如下图所示

这样的话计算 dp[1][3] 时,理论上dp[1][3] 应该等于1,但是实际上dp[1][3]值如下图结果为3:

原因就是对dp[0][0]错误地初始化为了3,正确地初始化dp[0][0]应该为0。

JS算法源码
/* JavaScript Node ACM模式 控制台输入获取 */ const readline = require("readline"); const rl = readline.createInterface({ input: process.stdin, output: process.stdout, }); const lines = []; rl.on("line", (line) => { lines.push(line); if (lines.length == 2) { const n = parseInt(lines[0]); const nums = lines[1].split(" ").map(Number); console.log(getResult(n, nums)); lines.length = 0; } }); function getResult(n, nums) { // 所有雨花石重量之和 const sum = nums.reduce((a, b) => a + b); // 如果重量之和不能整除2,则必然无法平分 if (sum % 2 != 0) return -1; // 背包承重 const bag = sum / 2; // 二维数组 const dp = new Array(n + 1).fill(0).map(() => new Array(bag + 1).fill(0)); // 初始化第一行,n是一个不可能的装满背包的物品数量 for (let i = 0; i <= bag; i++) dp[0][i] = n; dp[0][0] = 0; for (let i = 1; i <= n; i++) { const num = nums[i - 1]; for (let j = 1; j <= bag; j++) { if (j < num) { dp[i][j] = dp[i - 1][j]; } else { dp[i][j] = Math.min(dp[i - 1][j], dp[i - 1][j - num] + 1); } } } // 如果装满背包的最少物品数为n, 则说明没有平分方案,因为n个雨花石的重量之和为sumV,而背包的承重是bag = sumV // 2 if (dp[n][bag] == n) { return -1; } else { return Math.min(n - dp[n][bag], dp[n][bag]); } }
Java算法源码
import java.util.Arrays; import java.util.Scanner; public class Main { public static void main(String[] args) { Scanner sc = new Scanner(System.in); int n = Integer.parseInt(sc.nextLine()); int[] nums = Arrays.stream(sc.nextLine().split(" ")).mapToInt(Integer::parseInt).toArray(); System.out.println(getResult(n, nums)); } public static int getResult(int n, int[] nums) { // 所有雨花石重量之和 int sum = Arrays.stream(nums).sum(); // 如果重量之和不能整除2,则必然无法平分 if (sum % 2 != 0) return -1; // 背包承重 int bag = sum / 2; // 二维数组 int[][] dp = new int[n + 1][bag + 1]; // 初始化第一行,n是一个不可能的装满背包的物品数量 for (int i = 0; i <= bag; i++) { dp[0][i] = n; } // 2023.07.16 修改初始化问题 dp[0][0] = 0; for (int i = 1; i <= n; i++) { int num = nums[i - 1]; for (int j = 1; j <= bag; j++) { if (j < num) { dp[i][j] = dp[i - 1][j]; } else { dp[i][j] = Math.min(dp[i - 1][j], dp[i - 1][j - num] + 1); } } } // 如果装满背包的最少物品数为n, 则说明没有平分方案,因为n个雨花石的重量之和为sumV,而背包的承重是bag = sumV // 2 if (dp[n][bag] == n) { return -1; } else { return dp[n][bag]; } } }
Python算法源码
# 输入获取 n = int(input()) nums = list(map(int, input().split())) # 算法入口 def getResult(): # 所有雨花石重量之和 sumV = sum(nums) # 如果重量之和不能整除2,则必然无法平分 if sumV % 2 != 0: return -1 # 背包承重 bag = sumV // 2 # 二维数组 dp = [[0] * (bag + 1) for _ in range(n + 1)] # 初始化第一行,n是一个不可能的装满背包的物品数量 for i in range(bag + 1): dp[0][i] = n dp[0][0] = 0 for i in range(1, n + 1): num = nums[i - 1] for j in range(1, bag + 1): if j < num: dp[i][j] = dp[i - 1][j] else: dp[i][j] = min(dp[i - 1][j], dp[i - 1][j - num] + 1) # 如果装满背包的最少物品数为n, 则说明没有平分方案,因为n个雨花石的重量之和为sumV,而背包的承重是bag = sumV // 2 if dp[n][bag] == n: return -1 else: return min(n - dp[n][bag], dp[n][bag]) # 算法调用 print(getResult())
C算法源码
#include <stdio.h> #define MAX_SIZE 30 #define MAX_ROWS 30 + 1 #define MAX_COLS 30 * 1000 + 1 #define MIN(a,b) a < b ? a : b int getResult(int n, int* nums); int main() { int n; scanf("%d", &n); int nums[MAX_SIZE]; for(int i=0; i<n; i++) { scanf("%d", &nums[i]); } printf("%d\n", getResult(n, nums)); return 0; } int dp[MAX_ROWS][MAX_COLS] = {0}; int getResult(int n, int* nums) { // 所有雨花石重量之和 int sum = 0; for(int i=0; i<n; i++) { sum += nums[i]; } // 如果重量之和不能整除2,则必然无法平分 if(sum % 2 != 0) { return -1; } // 背包承重 int bag = sum / 2; // 二维数组 // 由于此处dp申请的内存过大,会造成栈内存溢出,因此将dp定义到全局 //int dp[MAX_ROWS][MAX_COLS] = {0}; // 初始化第一行,n是一个不可能的装满背包的物品数量 // 注意dp[0][0]保持0 for(int i=1; i<=bag; i++) { dp[0][i] = n; } for(int i=1; i<=n; i++) { int num = nums[i-1]; for(int j=1; j<=bag; j++) { if(j < num) { dp[i][j] = dp[i-1][j]; } else { dp[i][j] = MIN(dp[i-1][j], dp[i-1][j-num] + 1); } } } // 如果装满背包的最少物品数为n, 则说明没有平分方案,因为n个雨花石的重量之和为sumV,而背包的承重是bag = sumV // 2 if(dp[n][bag] == n) { return -1; } else { return dp[n][bag]; } }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/2/26 0:29:09

火速再就业!国安“救火教练”转身执教中甲保级队

北京时间12月23日&#xff0c;一则迅速发酵的消息引发关注&#xff1a;刚刚发文告别北京国安的代理主帅拉米罗&#xff0c;被曝将火速接手一支中甲保级球队——深圳青年人。此时&#xff0c;距离他在朋友圈那封充满感情的告别信发出&#xff0c;尚不足48小时。从首都豪门到南方…

作者头像 李华
网站建设 2026/2/20 6:44:16

个人理财收支记账系统 家庭理财系统APP_vj9n8--小程序论文

文章目录具体实现截图主要技术与实现手段系统设计与实现的思路系统设计方法java类核心代码部分展示结论源码lw获取/同行可拿货,招校园代理 &#xff1a;文章底部获取博主联系方式&#xff01;具体实现截图 本系统&#xff08;程序源码数据库调试部署讲解&#xff09;带文档1万…

作者头像 李华
网站建设 2026/2/25 6:22:18

零成本高效票务兑换操作指南:从渠道对接至流程落地全解析

1. 核心操作逻辑票务兑换的核心逻辑基于 “需求响应 - 渠道匹配 - 凭证流转” 的闭环&#xff0c;无需提前储备库存&#xff0c;也无需预先投入资金&#xff0c;仅通过对接正规票务供应渠道&#xff0c;在接收用户需求并确认付款后&#xff0c;从渠道获取对应的票务兑换凭证&am…

作者头像 李华
网站建设 2026/2/23 14:37:46

PPT打开密码,如何取消?

PPT文件打开的时候需要打开密码&#xff0c;这种情况该用什么方法解开打开密码呢&#xff1f; 如果是在知道密码的情况下&#xff0c;想要取消PPT文件的打开密码&#xff0c;可以这样做&#xff1a; 打开【文件】-【信息】-【保护演示文稿】&#xff0c;将输入框内的密码删除…

作者头像 李华
网站建设 2026/2/22 14:41:18

Docker 安装 MySQL 8.0

Docker 安装 MySQL 8.0 是日常运维中最常用的方式之一&#xff0c;核心是镜像拉取持久化配置适配 MySQL 8.0 特性&#xff08;如密码策略、认证插件、字符集等&#xff09;&#xff0c;以下是通用且避坑的详细步骤&#xff0c;适配 Linux&#xff08;CentOS/Debian/Ubuntu&…

作者头像 李华