news 2026/4/16 10:36:50

(新卷,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/4/15 11:43:50

(新卷,100分)- 单词加密(Java JS Python)

(新卷,100分)- 单词加密&#xff08;Java & JS & Python&#xff09;题目描述1、输入一个英文句子&#xff0c;句子中包含若干个单词&#xff0c;每个单词间有一个空格&#xff1b;2、需要将句子中的每个单词按照要求加密输出。要求&#xff1a;1&#xff09;单词中包括…

作者头像 李华
网站建设 2026/4/16 9:18:59

基于Android的校园二手跳蚤市场_ic9em-小程序-论文

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

作者头像 李华
网站建设 2026/4/15 8:51:23

基于Android的足疗公共浴池洗浴中心系统APP_xzt3v 小程序

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

作者头像 李华
网站建设 2026/4/14 13:45:31

2025最新!专科生必备10个AI论文工具:毕业论文写作全测评

2025最新&#xff01;专科生必备10个AI论文工具&#xff1a;毕业论文写作全测评 2025年专科生论文写作工具测评&#xff1a;为何需要这份榜单&#xff1f; 随着人工智能技术的不断进步&#xff0c;越来越多的AI工具开始进入学术领域&#xff0c;为学生和研究者提供高效的写作支…

作者头像 李华
网站建设 2026/4/13 10:32:34

基于Android的学生信息管理系统应用设计与实现(源码+文档)

包含项目报告&#xff0c;接近6200字数文档&#xff08;摘要、项目背景及意义、开发环境、开发技术、需求分析与可行性分析、数据库表设计、系统总体设计、实现关键代码&#xff0c;结论、参考文献&#xff09;&#xff1b;基于Android Studio开发软件已实现以下几个功能&#…

作者头像 李华