news 2025/12/31 18:33:49

数据结构之回溯算法

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
数据结构之回溯算法

一、回溯算法简介

回溯算法(Backtracking)是一种基于递归的穷举搜索算法,核心思想是 “尝试 - 回退 - 再尝试”:从初始状态出发,逐步探索所有可能的路径,当发现当前路径无法满足条件时(剪枝),撤销当前选择(回溯),回到上一步继续探索其他路径,直到找到解或遍历完所有可能。

回溯算法天然适配 “组合、排列、子集、切割、棋盘(如 N 皇后)” 等多选择、多约束的问题,是数据结构与算法中解决 “穷举类” 问题的核心方法。

二、核心要素

回溯算法的执行过程可类比 “走迷宫”:走一步→发现不通→退回来→换方向再走。其核心包含 4 个要素:

要素说明
路径已做出的选择(如已选的组合、排列元素,已放置皇后的位置)
选择列表当前步骤可选择的选项(如剩余未选的元素、皇后可放置的列)
结束条件到达决策树的叶子节点,路径满足要求(如组合长度达标、排列完成)
剪枝提前排除无效路径(如重复组合、皇后冲突),减少不必要的穷举(优化核心)

三、应用场景

以下通过 4 类经典问题,讲解回溯算法的具体实现,覆盖 “组合、排列、子集、棋盘” 四大核心场景。

场景 1:组合问题(无重复元素,不考虑顺序)

问题:给定数组nums = [1,2,3],找出所有长度为 2 的组合(如[1,2][1,3][2,3])。核心:组合不考虑顺序,需通过 “起始索引” 避免重复(如选 1 后只选 1 之后的元素)。

场景 2:排列问题(无重复元素,考虑顺序)

问题:给定数组nums = [1,2,3],找出所有全排列(如[1,2,3][1,3,2][2,1,3]等)。核心:排列考虑顺序,需通过 “已选集合” 避免重复选择同一元素。

场景 3:子集问题(所有可能的子集,包括空集)

问题:给定数组nums = [1,2,3],找出所有子集(如[][1][1,2][1,2,3][2]等)。核心:子集是 “选或不选” 的结果,结束条件可省略(每次递归都将当前路径加入结果)。

场景 4:棋盘问题(N 皇后,带复杂约束)

问题:N 皇后问题:在 N×N 的棋盘上放置 N 个皇后,使得任意两个皇后不在同一行、同一列、同一斜线,找出所有合法的放置方案。核心:通过剪枝快速排除无效位置(同行 / 同列 / 同斜线),减少穷举次数。

四、案例分享

题目:给定一个整型数组,其中所有元素都各不相同,返回这些元素所有可能的排列。

如[1,2,3],返回:[[1,2,3], [1,3,2], [2,1,3], [2,3,1], [3,1,2], [3,2,1]]。

import java.util.LinkedList; import java.util.List; public class Solution { List<List<Integer>> result = null; List<Boolean> used = null; // p中保存了一个有index个元素的排列,向这个排列末尾添加第index+1个元素 // 获得一个有index+1个元素的排列 void generatePermutation(int[] nums, int index, List<Integer> p) { if (index == nums.length) { result.add(new LinkedList<>(p)); return; } for (int i = 0; i < nums.length; ++i) if (!used.get(i)) { // 将nums[i]添加到p中 p.add(nums[i]); used.set(i, true); // 递归 generatePermutation(nums, index + 1, p); // 下面两行实现回溯,因为以后还会使用到nums[i],是逐个元素进行回溯 p.remove(p.size() - 1); used.set(i, false); } return; } // 46 使用递归和回溯的算法完成该题 public List<List<Integer>> permute(int[] nums) { result = new LinkedList<List<Integer>>(); if (nums.length == 0) return result; LinkedList<Integer> p = new LinkedList<>(); used = new LinkedList<>(); // 初始化used for (int i = 0; i < nums.length; ++i) used.add(i, false); generatePermutation(nums, 0, p); return result; } public static void main(String[] args) { int[] nums = { 1, 2, 3 }; System.out.println(new Solution().permute(nums)); } }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2025/12/28 10:52:07

46、深入探索编程符号、函数与操作:从基础到高级应用

深入探索编程符号、函数与操作:从基础到高级应用 1. 符号与运算符 在编程的世界里,各种符号和运算符是构建代码逻辑的基石。以下是一些常见符号及其用途: - 逻辑与比较运算符 : ! (非运算符)、 != (不等于)、 !~ (不匹配正则表达式)、 && (逻辑…

作者头像 李华
网站建设 2025/12/27 3:29:49

论AI时代下 “马扁” 子的趋势分析(一)

前言:问君能有几多愁,恰似一江春水向东流故事是这样的… 随着九紫离火大运拉开帷幕,愈演愈烈… 时间加速幻觉加重的背后,是对人性精心设计的一个个陷进,太多太多的痴男怨女,构成这副宏大的叙画. 不知觉中已深入局,立足根本,见真我… 北京的冬天,迎来2025年的第一场降雪,记忆中的…

作者头像 李华
网站建设 2025/12/30 22:17:57

7天拿下微软PowerBI证书真的太香了

&#x1f3af;微软认证&#xff1a;Power BI数据分析师助理&#xff0c;展示与使用 Microsoft Power BI 进行建模、可视化和分析数据的业务和技术要求相一致的方法和实践&#xff0c;是数据分析领域的敲门砖&#xff0c;特别适合想快速入门数据可视化工具的同学&#x1f49b;微…

作者头像 李华
网站建设 2025/12/31 0:33:19

JSP中如何设计大文件上传的交互界面与用户体验?

大文件上传系统开发指南&#xff08;基于原生JSSpringBoot&#xff09; 项目概述 大家好&#xff0c;我是一个在浙江奋斗的Java程序员&#xff0c;最近接了个"刺激"的外包项目 - 开发一个支持20G大文件上传下载的系统&#xff0c;还要兼容IE9这种上古浏览器。客户要…

作者头像 李华
网站建设 2025/12/31 11:39:48

wangEditor粘贴ppt幻灯片转存网页兼容处理

《.NET程序员的CMS升级记&#xff1a;Word一键粘贴公式全兼容&#xff0c;680元预算搞掂&#xff01;》 一、客户爸爸的需求 “小王啊&#xff0c;我们领导说每次从Word复制新闻到后台&#xff0c;表格变形、公式变乱码&#xff0c;连图片都丢了…” “张总&#xff0c;我调研…

作者头像 李华
网站建设 2025/12/29 23:12:18

从 paperxie 到工具矩阵:AI 开题报告工具如何帮你突破 “学术启动瓶颈”?

开题报告是学术研究的 “第一块拼图”—— 它需要你在有限时间内理清选题逻辑、匹配文献支撑、对齐格式规范&#xff0c;而单一工具往往难以覆盖全流程需求。从 paperxie 的 “流程化引导”&#xff0c;到其他工具的 “专项能力补位”&#xff0c;不同 AI 工具正在形成 “开题报…

作者头像 李华