
本文详细阐述了如何通过最少次数的列表反转(reverse)和旋转(rotate)操作,将一个整数列表转换成目标列表。文章采用递归深度优先搜索(dfs)策略,构建操作树,并引入父操作剪枝优化,避免重复计算。教程提供了java实现代码,涵盖了核心递归逻辑、列表操作辅助函数,以及如何高效地找出最短转换路径,并探讨了获取具体操作序列的方法。
1. 引言:列表转换问题概述
在软件开发中,我们经常需要对数据结构进行各种操作。设想这样一个场景:我们有两个整数列表a和b,它们包含相同的元素,但顺序不同。我们的目标是找出将列表a通过最少次数的特定操作转换成列表b所需的步骤。这里定义了两种核心操作:
反转 (Reverse):将列表中的所有元素顺序颠倒。例如,[1, 2, 3, 4] 反转后变为 [4, 3, 2, 1]。旋转 (Rotate):将列表的最后一个元素移动到列表的开头。例如,[1, 2, 3, 4] 旋转后变为 [4, 1, 2, 3]。
问题在于,从列表a到列表b可能存在多种操作序列,我们感兴趣的是其中操作次数最少的那一个。
2. 核心思路:基于递归的深度优先搜索
要找到最小操作数,我们可以将这个问题抽象为一个状态空间搜索问题。每个列表的当前状态都可以看作搜索树中的一个节点。从任何一个节点(即当前列表状态)出发,我们都可以应用reverse或rotate这两种操作,从而产生两个新的子节点(新的列表状态)。我们的目标就是从起始节点(列表a)出发,找到一条到达目标节点(列表b)的最短路径。
这种搜索过程非常适合使用递归的深度优先搜索(DFS)来实现。
立即学习“Java免费学习笔记(深入)”;
2.1 递归函数定义
我们将定义两个主要的函数:
minimumOps(List a, List b):这是对外暴露的入口函数,负责启动递归搜索。minimumOpsRec(List currentList, List targetList, int currentCount, OP lastOperation):这是一个辅助递归函数,用于执行实际的深度优先搜索。
3. 操作定义与辅助函数
为了进行列表操作,我们需要实现rotate和reverse方法。同时,为了在递归过程中追踪上一步的操作,我们定义一个枚举类型OP。
Revid AI
AI短视频生成平台
96 查看详情
import java.util.*;public class ListTransform { // 定义操作类型 enum OP { REV, // 反转 ROT // 旋转 } /** * 旋转列表:将最后一个元素移到开头。 * 注意:此方法返回一个新的列表,不修改原始列表。 * @param list 待旋转的列表 * @return 旋转后的新列表 */ private static List rotate(List list) { var newList = new ArrayList(list); // 使用 Collections.rotate(list, distance) 方法,distance 为 1 表示将最后一个元素移到开头 Collections.rotate(newList, 1); return newList; } /** * 反转列表:将列表中所有元素顺序颠倒。 * 注意:此方法返回一个新的列表,不修改原始列表。 * @param list 待反转的列表 * @return 反转后的新列表 */ private static List reverse(List list) { var newList = new ArrayList(list); Collections.reverse(newList); return newList; } // ... 递归搜索逻辑将在下一节实现}
重要提示: rotate 和 reverse 方法都创建并返回了一个新的列表,而不是直接修改传入的列表。这种“不可变性”在递归搜索中至关重要,它确保了每个递归调用都在一个独立的状态上操作,避免了副作用和状态混乱。
4. 递归搜索逻辑与剪枝优化
现在,我们来实现核心的递归搜索逻辑minimumOpsRec。
import java.util.*;public class ListTransform { enum OP { REV, ROT } private static List rotate(List list) { /* ... */ return list; } private static List reverse(List list) { /* ... */ return list; } /** * 启动列表转换的最小操作数计算。 * @param a 初始列表 * @param b 目标列表 * @return 最小操作数,如果无法转换则返回 Integer.MAX_VALUE */ public static int minimumOps(List a, List b) { // 如果初始列表和目标列表相同,则无需操作 if (Objects.equals(a, b)) return 0; // 分别尝试以反转和旋转开始,进行递归搜索 // 初始操作计数为1 int revCount = minimumOpsRec(reverse(a), b, 1, OP.REV); int rotCount = minimumOpsRec(rotate(a), b, 1, OP.ROT); // 返回两种起始操作中,操作数较小的那一个 return Math.min(revCount, rotCount); } /** * 递归辅助函数,用于深度优先搜索最小操作数。 * @param currentList 当前列表状态 * @param targetList 目标列表 * @param currentCount 当前已执行的操作数 * @param lastOperation 上一步执行的操作类型,用于剪枝优化 * @return 从 currentList 到 targetList 所需的最小额外操作数,加上 currentCount; * 如果无法转换或超出深度限制,则返回 Integer.MAX_VALUE。 */ public static int minimumOpsRec(List currentList, List targetList, int currentCount, OP lastOperation) { // 基本情况1:如果当前列表已达到目标列表,返回当前操作数 if (Objects.equals(currentList, targetList)) { return currentCount; } // 基本情况2:设置搜索深度限制。 // 如果操作数超过列表大小的某个倍数(例如,这里的 targetList.size() * 2), // 认为这条路径过长或无法达到目标,进行剪枝。 // 这个限制是一个启发式,可以防止无限递归,但对于某些复杂情况可能需要调整。 if (currentCount > targetList.size() * 2) { // 简单启发式剪枝 return Integer.MAX_VALUE; } // 递归调用,探索两种可能的后续操作 int revResult = Integer.MAX_VALUE; int rotResult; // 剪枝优化:如果上一步是反转操作,则避免再次立即反转。 // 因为 reverse(reverse(list)) == list,连续两次反转等同于没有操作, // 这种路径是冗余且低效的。 if (lastOperation != OP.REV) { revResult = minimumOpsRec(reverse(currentList), targetList, currentCount + 1, OP.REV); } // 总是可以尝试旋转操作 rotResult = minimumOpsRec(rotate(currentList), targetList, currentCount + 1, OP.ROT); // 返回两个分支中操作数较小的那一个 return Math.min(revResult, rotResult); } public static void main(String[] args) { var a = new ArrayList(List.of(1, 2, 3, 4)); var b = new ArrayList(List.of(2, 1, 4, 3)); // 目标:rotate(rotate(reverse(S))) var output = minimumOps(a, b); if (output == Integer.MAX_VALUE) { System.out.println("无法在合理步数内转换"); } else { System.out.println("最小转换次数: " + output); // 预期输出 3 } var c = new ArrayList(List.of(1, 2, 3, 4)); var d = new ArrayList(List.of(4, 2, 1, 3)); // 示例:无法通过此算法解决的组合 var output2 = minimumOps(c, d); if (output2 == Integer.MAX_VALUE) { System.out.println("列表 " + c + " 无法转换为 " + d + " (或超出深度限制)"); } }}
4.1 逻辑解析
入口函数 minimumOps:
检查 a 和 b 是否已相等,如果相等则返回 0。分别对 a 应用 reverse 和 rotate 操作,作为递归的起始点。调用 minimumOpsRec,初始操作计数为 1。返回两个分支中操作数较小的值。
递归函数 minimumOpsRec:
基本情况 1 (终止条件):如果 currentList 等于 targetList,说明我们找到了一个转换路径,返回当前的 currentCount。基本情况 2 (剪枝条件):如果 currentCount 超过了 targetList.size() * 2,我们认为这条路径太长,或者目标不可达,因此返回 Integer.MAX_VALUE。这个限制是防止无限递归和不必要的深层搜索的启发式方法。对于长度为 N 的列表,最坏情况下可能需要 N 次旋转才能使一个元素回到原位,反转操作也类似。N*2 是一个相对宽松的上限。递归调用与剪枝优化:if (lastOperation != OP.REV):这是核心的剪枝策略。如果上一步操作是 REV(反转),那么这一步就不再尝试 REV。因为连续两次反转操作(reverse(reverse(list)))会使列表回到原状,这是一种冗余且低效的路径。通过避免这种模式,可以显著减少搜索空间。rotate 操作总是可以尝试,因为它不会立即导致列表回到上一步的状态。结果合并:函数返回 revResult 和 rotResult 中的最小值,这确保了我们总是选择当前分支下的最短路径。
5. 扩展:获取操作序列
上述代码只能返回最小操作数。如果需要获取具体的转换操作序列(例如 [ROT, ROT, REV]),我们需要修改递归函数的返回值类型,使其包含操作列表。
import java.util.*;class ListTransformWithSequence { enum OP { REV, ROT } private static List rotate(List list) { var newList = new ArrayList(list); Collections.rotate(newList, 1); return newList; } private static List reverse(List list) { var newList = new ArrayList(list); Collections.reverse(newList); return newList; } /** * 启动列表转换的最小操作数计算,并返回操作序列。 * @param a 初始列表 * @param b 目标列表 * @return 包含最小操作数和操作序列的 Map.Entry。如果无法转换,操作数将是 Integer.MAX_VALUE。 */ public static Map.Entry<Integer, List> minimumOps(List a, List b) { if (Objects.equals(a, b)) { return new AbstractMap.SimpleEntry(0, new ArrayList()); } // 分别尝试以反转和旋转开始 Map.Entry<Integer, List> revResult = minimumOpsRec(reverse(a), b, 1, OP.REV); Map.Entry<Integer, List> rotResult = minimumOpsRec(rotate(a), b, 1, OP.ROT); // 比较两个结果,选择操作数更小的那个 if (rotResult.getKey() >= revResult.getKey()) { // 如果反转路径更短或相等,将当前操作(REV)添加到序列开头 revResult.getValue().add(0, OP.REV); return revResult; } else { // 如果旋转路径更短,将当前操作(ROT)添加到序列开头 rotResult.getValue().add(0, OP.ROT); return rotResult; } } /** * 递归辅助函数,用于深度优先搜索最小操作数和操作序列。 * @param currentList 当前列表状态 * @param targetList 目标列表 * @param currentCount 当前已执行的操作数 * @param lastOperation 上一步执行的操作类型 * @return 包含操作数和操作序列的 Map.Entry。 */ public static Map.Entry<Integer, List> minimumOpsRec(List currentList, List targetList, int currentCount, OP lastOperation) { // 基本情况1:如果当前列表已达到目标列表,返回当前操作数和空序列 if (Objects.equals(currentList, targetList)) { return new AbstractMap.SimpleEntry(currentCount, new ArrayList()); } // 基本情况2:深度限制 if (currentCount > targetList.size() * 2) { return new AbstractMap.SimpleEntry(Integer.MAX_VALUE, new ArrayList()); } Map.Entry<Integer, List> revBranchResult = null; Map.Entry<Integer, List> rotBranchResult; // 剪枝优化:避免连续两次反转 if (lastOperation != OP.REV) { revBranchResult = minimumOpsRec(reverse(currentList), targetList, currentCount + 1, OP.REV); } rotBranchResult = minimumOpsRec(rotate(currentList), targetList, currentCount + 1, OP.ROT); // 比较两个分支的结果 if (revBranchResult != null && rotBranchResult.getKey() >= revBranchResult.getKey()) { // 如果反转分支更优(或相等),将当前操作添加到其序列开头 revBranchResult.getValue().add(0, OP.REV); return revBranchResult; } else { // 如果旋转分支更优,将当前操作添加到其序列开头 rotBranchResult.getValue().add(0, OP.ROT);
以上就是Java中列表转换的最小操作数:递归搜索与优化策略的详细内容,更多请关注创想鸟其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1087631.html
微信扫一扫
支付宝扫一扫