算法通关村第十八关-青铜挑战回溯是怎么回事

news2025/1/19 14:31:05

大家好我是苏麟 , 今天聊聊回溯是怎么个事 .

回溯是最重要的算法思想之一,主要解决一些暴力枚举也搞不定的问题,例如组合、分割、子集、排列,棋盘等。从性能角度来看回溯算法的效率并不高,但对于这些暴力都搞不定的算法能出结果就很好了,效率低点没关系

我们利用LeetCode 77 组合题来了解回溯 . 77.组合

在这里插入图片描述

大纲

    • 从N叉树开始
    • 为什么有的问题暴力搜索也不行
    • 回溯 = 枚举 + 递归 + 撤销

回溯可以视为递归的拓展,很多思想和解法都与递归密切相关,在很多材料中都将回溯都与递归同时解释。因此学习回溯时,我们对比递归来分析其特征会理解更深刻。

  • 递归策略: 先与意中人制造偶遇,然后了解人家的情况,然后约人家吃饭,有好感之后尝试拉人家的手,没有拒绝就表白。
  • 回溯策略: 先统计周围所有的单身女孩,然后一个一个表白,被拒绝就说“我喝醉了”,然后就当啥也没发生,继续找下一个

回溯最大的好处是有非常明确的模板,所有的回溯都是一个大框架,因此透彻理解回溯的框架是解决一切回溯问题的基础

回溯不是万能的,而且能解决的问题也是非常明确的,例如组合、分割、子集、排列,棋盘等等,不过这些问题具体处理时又有很多不同

回溯模板 :
void huisu(参数){
	if(){
		return;
	}
	for(){
		处理......
		huisu();
		......
	}
}

从N叉树开始

在解释回溯之前,我们先看一下N叉树遍历的问题,我们知道在二又树中,按照前序遍历的过程如下所示 :


public class TreeNode {
    int val;
    TreeNode left;
    TreeNode right;
 }


public void nodeVal(TreeNode node){
        if(node == null){
             return;
         }
        System.out.println(node.val);
        nodeVal(node.left,list);
        nodeVal(node.right,list);
    }
}

假如我现在是一个三叉、四叉甚至N叉树该怎么办呢? 很显然这时候就不能用 left 和 right 来表示分支了,使用一个List比较好,也就是这样子 :

N 叉树的定义 :

class TreeNode{
	int val;
	List<TreeNode> nodes;
}

遍历的代码 :

public void nodeVal(TreeNode node){
        if(node == null){
             return;
         }
        System.out.println(node.val);
        for(int i = 1;i <= nodes.length;i++){
			nodeVal(第i个节点);
		}
    }
}

到这里,你有没有发现和上面说的回溯的模板非常像了? 是的!非常像!既然很像,那说明两者一定存在某种关系。其他暂时不管,现在你只要先明白回溯的大框架就是遍历N又树就行了。

为什么有的问题暴力搜索也不行

我们说回湖主要解决暴力枚举也解决不了的问题,什么问题这么神奇,暴力都搞不定?

LeetCode77: 给定两个整数 n 和 k,返回1…n 中所有可能的 k 个数的组合。
例如,输入n=4k=2,则输出 : [[1,2],[1,3],[1,4],[2,3],[2,4],[3,4]]

首先明确这个题是什么意思,如果n=4,k=2,那就是从4个数中选择2个,问你最后能选出多少组数据。这个是高中数学中的一个内容,过程大致这样: 如果n=4,那就是所有的数字为{1,2,3,4)

  • 1.先取一个1,则有[1,2],[1,3],[1,4]三种可能
  • 2.然后取一个因为1已经取过了,不再取,则有[2,3],[2,4]两种可能
  • 3.再取一个3,因为1和2都取过了,不再取,则有[3,4]一种可能
  • 4.再取4,因为1,2,3都已经取过了,所以直接返回null
  • 5.所以最终结果就是[1,2],[1,3],[1,4],[2,3],[2,4],[3,4]

这就是我们思考该问题的基本过程,写成代码也很容易,双层循环轻松搞定

int n = 4;        
 for (int i = 1; i <= n; i++) {  
 	for (int j = i + 1; j <= n; j++) {                
		System.out.println(i + " " + j);             
	}         
}

假如n和k都变大,比如n是200,k是3呢? 也可以,三层循环基本搞定

int n = 200;   
for (int i = 1; i <= n; i++) {        
	for (int j = i + 1; j <= n; j++) {             
		for (int u = j + 1; u <= n; n++) {                
			System.out.println(i + " " + j + " " + u);         
		}    
	}
}

如何这里的K是5呢? 甚至是50呢? 你需要套多少层循环? 甚至告你K就是一个末知的正整数k,你怎么写循环呢?这时候已经无能为例了? 所以暴力搜索就不行了。

这就是组合类型问题,除此之外子集、排列、切割、棋盘等方面都有类似的问题,因此我们要找更好的方式。

回溯 = 枚举 + 递归 + 撤销

我们继续研究LeetCode77题,我们图示一下上面自己枚举所有答案的过程。

n=4时,我们可以选择的n有 1,2,3,4这四种情况,所以我们从第一层到第二层的分支有四个,分别表示可以取1,2,3,4。而且这里 从左向右取数,取过的数,不在重复取。第一次取1,集合变为2,3,4,因为k为2,我们只需要再取一个数就可以了,分别取2,3,4,得到集合[1,2] [1,3][1,4],以此类推横向:

在这里插入图片描述
每次从集合中选取元素,可选择的范围会逐步收缩,到了取4时就直接为空了

继续观察树结构,可以发现,图中每次访问到一次叶子节点(图中红框标记处),我们就找到了一个结果。虽然最后一个是空,但是不影响结果。这相当于只需要把从根节点开始每次选择的内容(分支)达到叶子节点时,将其收集起来就是想要的结果。

如果感觉不明显,我们再画一个n=5,k=3的例子:

在这里插入图片描述
从图中我们发现元素个数n相当于树的宽度(横向),而每个结果的元素个数k相当于树的深度(纵向)。所以我们说回溯算法就是一纵一横而已。再分析,我们还发现几个规律:

  • 我们每次选择都是从类似{1,2,3,4),1,2,3,4,5这样的序列中一个个选的,这就是局部枚举,而且越往后枚举范围越小。
  • 枚举时,我们就是简单的暴力测试而已,一个个验证,能否满足要求,从上图可以看到,这就是N叉树遍历的过程,因此两者代码也必然很像。
  • 我们再看上图中红色大框起来的部分,这个部分的执行过程与n=4,k=2的处理过程完全一致,很明显这是个可以递归的子结构。

这样我们就将回溯与N叉树的完美结合在一起了

到此,还有一个大问题没有解决,回溯一般会有个手动撤销的操作,为什么要这样呢? 继续观察纵横图:
在这里插入图片描述

我们可以看到,我们收集每个结果不是针对叶子结点的,而是针对树枝的,比如最上层我们首先选了1,下层如果选2,结果就是(1,2),如果下层选了3,结果就是(1,3),依次类推。现在的问题是当我们得到第一个结果{1,2)之后,怎么得到第二个结果(1,3}呢?

继续观察纵横图,可以看到,我可以在得到(1,2)之后将2撤掉,再继续取3,这样就得到了(1,3},同理可以得到{1,4},之后当前层就没有了,我们可以将1撤销,继续从最上层取2继续进行。

这里对应的代码操作就是先将第一个结果放在临时列表 deque 里,得到第一个结果(1,2)之后就将path里的内容放进结果列表 list 中,之后,将 deque 里的2撤销掉,继续寻找下一个结果{1.3},然后继续将 deque 放入list,然后再撤销继续找。

这几条就是回溯的基本规律,明白之后,一切都变得豁然开朗。

到此我们就可以写出完整的回溯代码了 :

class Solution {
    public List<List<Integer>> combine(int n, int k) {
        List<List<Integer>> list = new ArrayList<>();
        if(n <= 0 || n < k){
            return list;
        }
        Deque<Integer> deque = new ArrayDeque<>();
        dfs(n,k,1,list,deque);
        return list;
    }
    //dfs 深度优先搜索的意思
    public void dfs (int n,int k,int start,List<List<Integer>> list,Deque<Integer> deque){
        if(deque.size() == k){
            list.add(new ArrayList<>(deque));
            return;
        }
        for(int i = start;i <= n;i++){
            deque.addLast(i);
            dfs(n,k,i + 1,list,deque);
            deque.removeLast();
        }
    }
}

上面代码还有个问题要解释一下: start 和 i 是怎么变化的,为什么传给下一层时要加 1.我们可以看到在递归里有个循环

for (int i = startIndex; i <= n; i++) {     
	dfs(n,k,i+1,path,res);  
}

这里的循环有什么作用呢? 看一下图就知道了,这里其实就是枚举,第一次n=4,可以选择1,2,3,4四种情况,所以就有四个分支,for循环就会执行四次:

在这里插入图片描述
而对于第二层第一个,选择了1之后,剩下的元素只有2,3,4了,所以这时候for循环就执行3次,后面的则只有2次和1次。

这期就到这里 , 下期见!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1293569.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

快速排序的非递归实现

上期我们实现了快速排序的递归实现&#xff0c;但是我们知道如果递归深度太深&#xff0c;栈就会溢出&#xff0c;所以我们本期将为大家讲述快速排序的非递归实现&#xff0c;我们需要用到栈的数据结构&#xff0c;我们知道栈中的数据全是在堆区开辟的空间&#xff0c;堆的空间…

【EXCEL】offset函数

语法&#xff1a; offset(reference,row,column,[height],[width]) 例子&#xff1a;

【日常总结】mybatis-plus WHERE BINARY 中文查不出来

目录 一、场景 二、问题 三、原因 四、解决方案 五、拓展&#xff08;全表全字段修改字符集一键更改&#xff09; 准备工作&#xff1a;做好整个库备份 1. 全表一键修改 Stage 1&#xff1a;运行如下查询 Stage 2&#xff1a;复制sql语句 Stage 3&#xff1a;执行即可…

Volumetric Lights 2 HDRP

高清晰度渲染管道,包括先进的新功能,如半透明阴影图和直接灯光投射加上许多改进。 插件是一个快速,灵活和伟大的前瞻性光散射解决方案的高清晰度渲染管道。只需点击几下,即可改善场景中的照明视觉效果。 兼容: 点光源 聚光灯 碟形灯 矩形灯 通过覆盖摄像机周围大面积区域的…

05 JQuery基础入门

文章目录 一、jQuery介绍1. 简介2. 版本介绍3. 相关网站4. HTML引入方式 二、基础语法1. 顶级对象$2. 与DOM对象转化3. 选择器4. 事件5. 动画6. 修改样式7. 修改属性 一、jQuery介绍 1. 简介 jQuery是JavaScript编程语言底层库&#xff0c;它是一个快速&#xff0c;简洁的Jav…

JavaScript <关于逆向RSA非对称加密算法的案例(附原代码)>--案例(五)

前言: 趁热打铁,标记一下RSA的算法逆向...第二篇会有详解(本篇重在过程) 正文: 废话不说,直接分析步骤图: 到了这里,可以看到在登录的时候,需要验证码(本篇不教反验证码) 下面是正题--->逆他的pwd(密码) 总结: 问题:怎么确定一个密文数据是基于什么算法做出来的呢? 答:…

python 涉及opencv mediapipe知识,眨眼计数 供初学者参考

基本思路 我们知道正面侦测到人脸时&#xff0c;任意一只眼睛水平方向上的两个特征点构成水平距离&#xff0c;上下两个特征点构成垂直距离 当头像靠近或者远离摄像头时&#xff0c;垂直距离与水平距离的比值基本恒定 根据这一思路 当闭眼时 垂直距离变小 比值固定小于某一个…

Abaqus基础教程--胶合失效仿真

胶合是电子行业中常见的连接方式&#xff0c;abaqus中常用cohesive单元或者cohesive接触两种方法进行胶合失效仿真&#xff0c;这两种方式操作方法有所差别&#xff0c;但结果一般大同小异。 本例模型比较简单&#xff0c;建模过程从略&#xff0c;使用静态分析&#xff0c;使…

python基于ModBusTCP服务端的业务实现特定的client

python实现ModBusTCP协议的client是一件简单的事情&#xff0c;只要通过pymodbus、pyModbusTCP等模块都可以实现&#xff0c;本文采用pymodbus。但要基于ModBusTCP服务端的业务实现特定的client&#xff0c;那得看看服务端是否复杂。前面系列文章&#xff0c;我们学习了对服务端…

【vtkWidgetRepresentation】第五期 vtkLineRepresentation

很高兴在雪易的CSDN遇见你 内容同步更新在公众号“VTK忠粉” 【vtkWidgetRepresentation】第五期 一条直线的交互 前言 本文分享vtkLineRepresentation&#xff0c;希望对各位小伙伴有所帮助&#xff01; 感谢各位小伙伴的点赞关注&#xff0c;小易会继续努力分享&#xf…

Python---继承

1、什么是继承 我们接下来来聊聊Python代码中的“继承”&#xff1a;类是用来描述现实世界中同一组事务的共有特性的抽象模型&#xff0c;但是类也有上下级和范围之分&#xff0c;比如&#xff1a;生物 > 动物 > 哺乳动物 > 灵长型动物 > 人类 > 黄种人 从哲学…

Go--协程

协程 协程是Go语言最大的特色之一。 1、协程的概念 协程并不是Go发明的概念&#xff0c;支持协程的变成语言有很多。Go在语言层面直接提供对协程的支持称为goroutine。 1.1 基本概念 进程 进程是应用程序启动的实例&#xff0c;每个进程都有独立的内存空间&#xff0c;不同…

DSP外部中断笔记

中断原理 三部分 注意 &#xff0c;外部中断使能&#xff0c;PIE使能&#xff0c;CPU中断使能 外部中断有7个&#xff0c;PIE有12组&#xff0c;一个组有8个中断复用。只有一个CPU中断可执行。 外部中断原理 1、外部中断概述 外部中断结构图 外部中断XINT1对应的是0到31GPI…

<IBM Websphere Portal>《关于IBM的Portal和WAS的说明和总结(自用笔记)》

《关于IBM的Portal和WAS的简单总结》 1 架构1.1 说明 2 常见问题2.1 LDAP链接问题2.2 启动脚本建议2.3 日志大小保留建议2.4 启动垃圾回收日志 3 日志位置 1 架构 应用服务部署架构如上&#xff1a; &#x1f449;192.168.66.1服务器运行的server进程有&#xff1a;dmgr、nodea…

360压缩安装一半不动了怎么办?

360压缩软件是我们常用的压缩软件&#xff0c;但是常常会遇到压缩安装到一半停止的情况&#xff0c;下面提供了一些可能的原因和解决办法&#xff0c;大家可以进行尝试~ 方法一&#xff1a;关闭防火墙和杀毒软件 有时候&#xff0c;防火墙和杀毒软件可能会阻止360压缩的安装过…

为什么 SQL 不适合图数据库

背景 “为什么你们的图形产品不支持 SQL 或类似 SQL 的查询语言&#xff1f;” 过去&#xff0c;我们的一些客户经常问这个问题&#xff0c;但随着时间的推移&#xff0c;这个问题变得越来越少。 尽管一度被忽视&#xff0c;但图数据库拥有无缝设计并适应其底层数据结构的查询…

Docker实战笔记 二 Springboot Idea 插件打包

1.上传springboot的jar rootcenots-7.5:/home/code#rz -----app.jar 2.编辑Dockerfile rootcenots-7.5:/home/code#vi Dockerfile内容 FROM openjdk:8 # 作者 MAINTAINER nnd # 声明要使用的端口 EXPOSE 8080 # VOLUME 指定了临时文件目录为/tmp。# 将本地包添加到容器中并…

服装收银系统哪个最好用

服装订货系统哪个最好&#xff0c;可能没有一个标准的答案&#xff0c;但至少可以从以下几点进行选择&#xff1a; 1、数据批量操作&#xff1a;服装到货都是一批一批&#xff0c;如果能将条码进行批量导入&#xff0c;这样在这里耗去的时间就少很多了&#xff0c;剩下的是时间…

在Windows 11中更改文件的扩展名有几种办法,个别办法可以批量修改

本文介绍了如何在Windows 11中更改文件的文件扩展名。 用简单的方法更改文件扩展名 对于大多数人来说&#xff0c;在Windows 11中更改文件扩展名的最简单方法是在更改文件名的同一个地方进行更改。然而&#xff0c;Windows默认情况下不显示文件扩展名&#xff0c;所以在我们可…

【Flink系列三】数据流图和任务链计算方式

上文介绍了如何计算并行度和slot的数量&#xff0c;本文介绍Flink代码提交后&#xff0c;如何生成计算的DAG数据流图。 程序和数据流图 所有的Flink程序都是由三部分组成的&#xff1a;Source、Transformation和Sink。Source负责读取数据源&#xff0c;Transformation利用各种…