把一个数组分为两部分使得其和相差最小——Java实现(详细注释,全面解析)

本文主要是介绍把一个数组分为两部分使得其和相差最小——Java实现(详细注释,全面解析),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

package bag;public class Test {/*** 给一个整数数组 array,把他分为两部分 a 和 b* 使得 a 和 b 的元素之和的差最小** @param array 输入数组* @return 返回差值最小是多少* <p>* 解析:array分出来的两个数组a、b之和sum1、sum2等于总和SUM* 既然要 sum1 与 sum2 之间相差最小则他们应该尽量接近 SUM/2* <p>* 即问题转变为:在array中取出数据存入a中,在不超过 SUM/2 的情况下使得a的总和最大* <p>* 也就是0-1背包问题*/public static int getTwoSubArrayMinDiff(int[] array) {int ans = 0;if (array.length == 0)return ans;int n = array.length;int SUM = 0;for (int i = 0; i < n; i++) {SUM += array[i];}// 状态转移方程:设待选的各个数据为 i,容量为 SUM/2,// dp[i][c] = max( dp[i-1][c], dp[i-1][C-c[i]] + c[i])/*解释:c 就是 capacity,每次从array里面取走一个数据,存在: 放入新数组a里面和不放入新数组里面a 两个选择不管是否放入i值都少了一个:i-1* 放入的时候:不仅要给 i 值减一,且将a中剩余可存入的数量减少响应值,即 C-c[i] (总容量减去已存入的等于剩余的)* 不放入的时候:仅需将i减一即可那么以上两种情况选择哪一种会导致a的和最接近 SUM/2 呢?使用max取本次dp[i][c]的最大值即可*/int[][] dp = new int[n + 1][SUM / 2 + 1];/*本题并不完全是一个背包问题,这里没有价值只有重量,也就是每一个数组元素的值。外层循环从0开始也行从1开始也行,注意别越界就可以了,内层循环每来一次,就得重新计算 dp[i][1...C] 的值,内层循环的含义是:每次从array里面拿到一个数据后要考虑是否放到新数组a里面,那么就得计算该元素放入是否能加入到a以及能否获取最大收益。最终不管是否加入了a,该元素都不再考虑。那么内层循环里面通过不断改变背包的容量(从1到SUM/2)来确定是否能产生最大收益*/for (int i = 0; i < n; i++) {for (int j = 1; j <= SUM / 2; j++) { // 假设当前的背包容量是 j// 这是不放入背包的情况dp[i + 1][j] = dp[i][j];// 这是放入背包的情况if (array[i] <= j // 先要确保array[i]的值不能大于背包容量&& dp[i][j - array[i]] + array[i] > dp[i][j]) { // 然后可以放心的把当前物品放入背包dp[i + 1][j] = dp[i][j - array[i]] + array[i];}// dp[i + 1][j] 会得到一个最大值}}// dp[n][SUM / 2] 是新数组a的和,他的两倍应该近似于SUM,差值即为两个新数组a、b之间的和最小差return SUM - 2 * dp[n][SUM / 2];}public static void main(String[] args) {
//        int[] array = {2, 3, 4, 6};
//        int[] array = {5, 6, 8, 7};int[] array = {25, 0, 18, 3, 9, 4, 12, 7, 6, 11};System.out.println(getTwoSubArrayMinDiff(array));System.out.println(getTwoSubArrayMinDiff2(array));}public static int getTwoSubArrayMinDiff2(int[] array) {if (array.length == 0) {return 0;}int sum = 0;for (int x : array) {sum += x;}int n = array.length;// 状态转移方程int[][] dp = new int[n][sum / 2 + 1];// 首先检查 0 号元素是否可以存入背包for (int i = 0; i <= sum / 2; i++ ) {dp[0][i] = (i >= array[0]) ? array[0] : 0;}// 然后再动态规划处理for (int i = 1; i < n; i++) {for (int j = 0; j <= sum / 2; j++) {dp[i][j] = dp[i - 1][j];// 若剩余空间可以存入当前的array[i]if (j >= array[i]) {dp[i][j] = Math.max(dp[i][j], dp[i - 1][j - array[i]] + array[i]);}}}return sum - 2 * dp[n - 1][sum / 2];}
}

这篇关于把一个数组分为两部分使得其和相差最小——Java实现(详细注释,全面解析)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1004002

相关文章

C++对象布局及多态实现探索之内存布局(整理的很多链接)

本文通过观察对象的内存布局,跟踪函数调用的汇编代码。分析了C++对象内存的布局情况,虚函数的执行方式,以及虚继承,等等 文章链接:http://dev.yesky.com/254/2191254.shtml      论C/C++函数间动态内存的传递 (2005-07-30)   当你涉及到C/C++的核心编程的时候,你会无止境地与内存管理打交道。 文章链接:http://dev.yesky

揭秘未来艺术:AI绘画工具全面介绍

📑前言 随着科技的飞速发展,人工智能(AI)已经逐渐渗透到我们生活的方方面面。在艺术创作领域,AI技术同样展现出了其独特的魅力。今天,我们就来一起探索这个神秘而引人入胜的领域,深入了解AI绘画工具的奥秘及其为艺术创作带来的革命性变革。 一、AI绘画工具的崛起 1.1 颠覆传统绘画模式 在过去,绘画是艺术家们通过手中的画笔,蘸取颜料,在画布上自由挥洒的创造性过程。然而,随着AI绘画工

Java五子棋之坐标校正

上篇针对了Java项目中的解构思维,在这篇内容中我们不妨从整体项目中拆解拿出一个非常重要的五子棋逻辑实现:坐标校正,我们如何使漫无目的鼠标点击变得有序化和可控化呢? 目录 一、从鼠标监听到获取坐标 1.MouseListener和MouseAdapter 2.mousePressed方法 二、坐标校正的具体实现方法 1.关于fillOval方法 2.坐标获取 3.坐标转换 4.坐

Spring Cloud:构建分布式系统的利器

引言 在当今的云计算和微服务架构时代,构建高效、可靠的分布式系统成为软件开发的重要任务。Spring Cloud 提供了一套完整的解决方案,帮助开发者快速构建分布式系统中的一些常见模式(例如配置管理、服务发现、断路器等)。本文将探讨 Spring Cloud 的定义、核心组件、应用场景以及未来的发展趋势。 什么是 Spring Cloud Spring Cloud 是一个基于 Spring

Javascript高级程序设计(第四版)--学习记录之变量、内存

原始值与引用值 原始值:简单的数据即基础数据类型,按值访问。 引用值:由多个值构成的对象即复杂数据类型,按引用访问。 动态属性 对于引用值而言,可以随时添加、修改和删除其属性和方法。 let person = new Object();person.name = 'Jason';person.age = 42;console.log(person.name,person.age);//'J

java8的新特性之一(Java Lambda表达式)

1:Java8的新特性 Lambda 表达式: 允许以更简洁的方式表示匿名函数(或称为闭包)。可以将Lambda表达式作为参数传递给方法或赋值给函数式接口类型的变量。 Stream API: 提供了一种处理集合数据的流式处理方式,支持函数式编程风格。 允许以声明性方式处理数据集合(如List、Set等)。提供了一系列操作,如map、filter、reduce等,以支持复杂的查询和转

解析 XML 和 INI

XML 1.TinyXML库 TinyXML是一个C++的XML解析库  使用介绍: https://www.cnblogs.com/mythou/archive/2011/11/27/2265169.html    使用的时候,只要把 tinyxml.h、tinystr.h、tinystr.cpp、tinyxml.cpp、tinyxmlerror.cpp、tinyxmlparser.

Java面试八股之怎么通过Java程序判断JVM是32位还是64位

怎么通过Java程序判断JVM是32位还是64位 可以通过Java程序内部检查系统属性来判断当前运行的JVM是32位还是64位。以下是一个简单的方法: public class JvmBitCheck {public static void main(String[] args) {String arch = System.getProperty("os.arch");String dataM

详细分析Springmvc中的@ModelAttribute基本知识(附Demo)

目录 前言1. 注解用法1.1 方法参数1.2 方法1.3 类 2. 注解场景2.1 表单参数2.2 AJAX请求2.3 文件上传 3. 实战4. 总结 前言 将请求参数绑定到模型对象上,或者在请求处理之前添加模型属性 可以在方法参数、方法或者类上使用 一般适用这几种场景: 表单处理:通过 @ModelAttribute 将表单数据绑定到模型对象上预处理逻辑:在请求处理之前

eclipse运行springboot项目,找不到主类

解决办法尝试了很多种,下载sts压缩包行不通。最后解决办法如图: help--->Eclipse Marketplace--->Popular--->找到Spring Tools 3---->Installed。