当前位置: 首页 > wzjs >正文

手机网站建设公司联系电话厦门网站个人制作

手机网站建设公司联系电话,厦门网站个人制作,软件下载免费app大全,wordpress 翻译更新在学习和使用回溯算法解决问题时&#xff0c;我们经常会遇到需要维护一个结果列表&#xff0c;例如所有可能的子集、组合或排列。 这个结果列表通常是一个 List<List<Integer>>&#xff0c;其中内部的 List<Integer> 代表一个具体的解。 然而&#xff0c;在…

在学习和使用回溯算法解决问题时,我们经常会遇到需要维护一个结果列表,例如所有可能的子集、组合或排列。 这个结果列表通常是一个 List<List<Integer>>,其中内部的 List<Integer> 代表一个具体的解。

然而,在构建这些内部的 List<Integer> 时,我们应该使用 List 接口还是 ArrayList 类呢? 这个问题看似简单,但背后隐藏着一个关于深拷贝和浅拷贝的重要概念,它直接影响到回溯算法的正确性。

问题重现:回溯算法中的陷阱

让我们考虑一个简单的例子:使用回溯算法找到一个数组的所有子集。 一个常见的实现方式如下:

import java.util.ArrayList;
import java.util.List;public class Subsets {public static List<List<Integer>> subsets(int[] nums) {List<List<Integer>> res = new ArrayList<>();backtrack(nums, 0, new ArrayList<>(), res);return res;}private static void backtrack(int[] nums, int index, List<Integer> currentSubset, List<List<Integer>> res) {// 基本情况:将当前子集添加到结果列表res.add(currentSubset); // 潜在的问题!// 递归探索for (int i = index; i < nums.length; i++) {currentSubset.add(nums[i]); // 选择backtrack(nums, i + 1, currentSubset, res);currentSubset.remove(currentSubset.size() - 1); // 撤销选择 (回溯)}}public static void main(String[] args) {int[] nums = {1, 2, 3};List<List<Integer>> allSubsets = subsets(nums);System.out.println(allSubsets);}
}

展开

这段代码看起来很合理,但如果你运行它,你会发现结果是错误的! 例如,对于输入 [1, 2, 3],你可能会得到类似 [[], [], [], [], [], [], [], []] 的结果,而不是预期的 [[], [1], [2], [1, 2], [3], [1, 3], [2, 3], [1, 2, 3]]

问题出在哪里?

问题在于这行代码:

res.add(currentSubset);

这里,我们直接将 currentSubset 添加到 res 中,而没有进行任何拷贝。 这意味着 res 中的所有元素都指向同一个 currentSubset 对象。 当我们在回溯过程中修改 currentSubset 时,res 中的所有元素都会受到影响,最终导致错误的结果。

深拷贝的必要性

为了解决这个问题,我们需要对 currentSubset 进行深拷贝,然后再将其添加到 res 中。 深拷贝会创建一个新的 ArrayList 对象,并将 currentSubset 中的所有元素复制到这个新的 ArrayList 中。 这样,res 中的每个元素都将指向一个独立的列表,而对 currentSubset 的修改不会影响到 res 中的其他元素。

正确的代码如下:

import java.util.ArrayList;
import java.util.List;public class Subsets {public static List<List<Integer>> subsets(int[] nums) {List<List<Integer>> res = new ArrayList<>();backtrack(nums, 0, new ArrayList<>(), res);return res;}private static void backtrack(int[] nums, int index, List<Integer> currentSubset, List<List<Integer>> res) {// 基本情况:将当前子集添加到结果列表res.add(new ArrayList<>(currentSubset)); // 深拷贝!// 递归探索for (int i = index; i < nums.length; i++) {currentSubset.add(nums[i]); // 选择backtrack(nums, i + 1, currentSubset, res);currentSubset.remove(currentSubset.size() - 1); // 撤销选择 (回溯)}}public static void main(String[] args) {int[] nums = {1, 2, 3};List<List<Integer>> allSubsets = subsets(nums);System.out.println(allSubsets);}
}

展开

现在,res.add(new ArrayList<>(currentSubset)) 创建了一个 currentSubset 的副本,并将这个副本添加到 res 中。 这样,res 中的每个元素都指向一个独立的列表,结果就是正确的。

http://www.dtcms.com/wzjs/805560.html

相关文章:

  • 风景区网站建设项目建设可行性wordpress php推送
  • 住房和城市建设部网站做网站视频背景
  • 移动端网站和app开发凤冈建设局网站
  • c 做网站房产信息网的官网链接
  • 淮安做网站就找卓越凯欣建筑工程有限责任公司
  • 南京关键词网站排名对百度竞价排名的看法
  • 网站建设推广选哪家新手网站建设教程
  • 网站推广的常用方法有哪些?莱芜摩托车网站
  • 怎么做网站滑动图片部分h5大人室内设计网
  • 城阳网站制作微小店网站建设官网
  • 郑州 (网站建设网站开发工程师发展趋势
  • 网站负责人备案采集照小型网上商城系统
  • 做彩票网站是违法吗如何建网站做传奇网友
  • 建设网站要什么电脑做珠宝网站公司
  • 邯郸单位网站建设网站多少图片怎么做超链接
  • 专门做婚姻法的网站四川建设人才网官网
  • 有源码怎么在本地网站搭建无锡做食品网站的公司哪家好
  • ps兼职做网站微信小程序源码网站
  • 公众号视频网站怎么做有没有什么东西可以做网站
  • 佛山市品牌网站建设价格泰安东平房产信息网
  • 风险网站怎么解决方案合肥官网建设公司哪家好
  • 南宁百度网站推广ueeshop外贸建站公司
  • iis做网站视毕节建设公司网站
  • 地方网站如何做北京做网站的好公司
  • 个人网站模板制作运城做网站
  • 简约式网站做个网站大概多少钱
  • 自己做网站怎么租服务器2021企业公司大黄页
  • 做美团网站需要多少钱唐山企业网站模板建站
  • 苏州网站推广优化如何做自己的简历网站
  • 加拿大购物网站排名网站推广seo代理