回溯理论基础及leetcode例题

学习参考

回溯

与递归相辅相成;回溯是递归的副产品,只要有递归就会有回溯。
回溯函数也就是递归函数,指的都是一个函数。

回溯搜索法

纯暴力搜索
解决的问题

组合问题:N个数里面按一定规则找出k个数的集合
切割问题:一个字符串按一定规则有几种切割方式
子集问题:一个N个数的集合里有多少符合条件的子集
排列问题:N个数按一定规则全排列,有几种排列方式(与组合差别,排列有元素顺序)
棋盘问题:N皇后,解数独等等

理解

抽象的不易理解;抽象为图形结构--树形结构
N叉树【树的宽度:集合的大小(for处理);深度:递归的深度(递归处理)】

模板

void backtracking(参数){
  if(终止条件){
    收集结果;
    return;
  }

  //单层搜索
   for(选择:本层集合中元素(树中节点孩子的数量就是集合的大小)){//集合元素集
      处理节点;
      backtracking(路径,选择列表);//递归函数;
      回溯操作; //(12,把2回溯,变13;没有回溯操作就会递归为123)
    }
  return;
}

递归里面嵌套for循环,for循环里又有递归

leetcode题目

组合

77.组合

for循环嵌套太多层了

树形结构
回溯理论基础及leetcode例题
不能取前面的的:因为组合是无序的,会重复;
每个节点都是一个for循环

回溯三部曲

递归函数参数返回值
确定终止条件
单层递归逻辑

伪代码

全局变量:二维数组res【返回值】
         一维数组path【单个结果】
//确定返回值参数
void backtracking(n,k,start){//n集合大小;k需要的子集合大小;start每个取值的开始;
  //确定终止条件
  if(path.size == k){
    res.add(path);
    return;
  }
  //单层递归逻辑
  //对于1,234节点
  for(i=start,i<=n;i++){
    path.push(i);//1
    backtracking(n,k,i+1);//遍历剩下的集合234;
    path.pop();//回溯过程
  } 
}

实现
java版本

class Solution {
    List<List<Integer>> res = new ArrayList<List<Integer>>();
    List<Integer> path = new ArrayList<Integer>();

    public List<List<Integer>> combine(int n, int k) {
        backtracking(n,k,1);
        return res;
    }

    public void backtracking(int n,int k,int start){
        if(path.size() == k){
            res.add(new ArrayList<>(path));//容易犯错误
            return;
        }

        for(int i=start;i<=n;i++){//i<=n -(k-path.size()) + 1 会减少运行时间【剪枝操作】
            path.add(i);
            backtracking(n,k,i+1);
            path.remove(path.size()-1);
        }
        
    }
}

问题:参考
在链表path里面添加值,然后把path链表添加进res链表中,在做算法题的时候,平时使用res.add(path),结果发现输出打印为空:

在链表path里面添加值,然后把path链表添加进res链表中,在做算法题的时候,平时使用res.add(path),结果发现输出打印为空: res.add(new ArrayList<>(path))和res.add(path)的区别
共同点: 都是向res这个ArrayList中填加了一个名为path的链表
不同点: res.add(new ArrayList(path)):开辟一个独立地址,地址中存放的内容为path链表,后续path的变化不会影响到res
res.add(path):将res尾部指向了path地址,后续path内容的变化会导致res的变化。

优化:剪枝
可以剪枝的地方就在递归中每一层的for循环所选择的起始位置。

如果for循环选择的起始位置之后的元素个数 已经不足 我们需要的元素个数了,那么就没有必要搜索了。

回溯理论基础及leetcode例题
优化过程如下:

已经选择的元素个数:path.size();
所需需要的元素个数为: k - path.size();
列表中剩余元素(n-i) >= 所需需要的元素个数(k - path.size())
在集合n中至多要从该起始位置 : i <= n - (k - path.size()) + 1,开始遍历

原文链接:https://www.cnblogs.com/yunshalee/p/17312416.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:回溯理论基础及leetcode例题 - Python技术站

(0)
上一篇 2023年4月17日
下一篇 2023年4月17日

相关文章

  • python 随机森林算法及其优化详解

    下面是详细讲解“Python随机森林算法及其优化详解”的完整攻略。 随机森林算法 随机森林是一种集成学习算法,是由多个决策树组成的。随机森林的基本思是通过对多个决策树的预测结果进行综合,来得到更加准确的预测结果。 随机森林算法的主要骤如下: 从原始数据集中随机选择一定数量的样本,建一个训练集。 随机选择一定数量特征,构建一个决树。 重复步骤1和步骤2,构建多…

    python 2023年5月14日
    00
  • C语言编程简单却重要的数据结构顺序表全面讲解

    C语言编程简单却重要的数据结构顺序表全面讲解 什么是顺序表? 顺序表是一种线性表,指的是一组有限元素的有限序列,其元素的逻辑顺序与它们在分配到的内存地址上的物理顺序相同或者等价。也就是说,顺序表中的元素按照其在内存中的位置依次存放。 顺序表的实现方式 顺序表的实现方式一般是使用数组,数组中的每一个元素对应着顺序表中的一个元素,位置相对应。 顺序表的优点 支持…

    数据结构 2023年5月17日
    00
  • Python实例详解递归算法

    下面是关于“Python实例详解递归算法”的完整攻略。 1. 递归算法概述 递归算法是一种基于函数调用自身的算法,它的基本思想是将一个大问题分解成若干个小问题,然后递归地解决每个小问题,最终将所有小问题的解合并成大问题的解。在Python中,我们可以使用递归算法来解决各种问题,例如计算阶乘、斐波那契数列等。 2. 递归算法实现 2.1 计算阶乘 阶乘是一个正…

    python 2023年5月13日
    00
  • Python实现决策树C4.5算法的示例

    Python实现决策树C4.5算法的示例 什么是决策树C4.5算法? 决策树C4.5算法是一种常用的分类算法,它的基思通过对数据集进行划分,构建一棵树形结构,从而实现对数据的分类。C4.5算法是ID3算法改进版,它在ID3算法的基础上引入了信息增益比的概念,解决了ID3算法中存在的一些问题。 决策树C4.5算法的实现步骤 决策树C4.5算法的实现步骤如下: …

    python 2023年5月14日
    00
  • Python算法之图的遍历

    下面是关于“Python算法之图的遍历”的完整攻略。 1. 图的遍历简介 图的遍历是指从图的某个顶点出发,按照一定的规则依访问图中的顶点,且每个点仅被访问一次的过程。图的遍历算法是图论中的基本算法一,常用于解决图论中一些问题,如最短路径、连通性等。 2 Python实现图的遍历 2.1 算法流程 图遍历算法主要有两种:深度优先遍历(DFS和广度优先遍历(BF…

    python 2023年5月13日
    00
  • Python迭代用法实例教程

    下面是详细讲解“Python迭代用法实例教程”的完整攻略。 1. 什么是迭代 迭代是指重复执行一组操作,直到满足特定条件为止。在Python中,迭代常用于遍历序列(列表、元组、字符串等)或其他可迭代对象(如字典、集合等)中的元素。 2. 迭代器和可迭代对象 在Python中,迭代器是一种可以遍历序列或其他可迭代对象的对象。迭代器对象可以使用next()函数来…

    python 2023年5月14日
    00
  • C语言一篇精通链表的各种操作

    C 语言精通链表操作攻略 简介 链表是一种常用的数据结构,它相对于数组等数据结构来说,具有更高的灵活性和增删操作的效率。在 C 语言中,我们可以用指针来实现链表,这也是指针与动态内存分配的重要应用之一。本文将提供在 C 语言中精通链表操作的攻略,包括链表的创建、添加、删除、搜索、遍历等常用操作。 基本结构体定义 链表一般由结构体和指针构成,下面我们定义一个链…

    数据结构 2023年5月17日
    00
  • 一文教你用python编写Dijkstra算法进行机器人路径规划

    一文教你用Python编写Dijkstra算法进行机器人路径规划 Dijkstra算法是一种用于寻找图中最短路径的算法,它的基本思想是从起点开始逐步扩展到离起点越来越远的节点,直到到达终点为止。在这个过程中,我们维护一个距,用于记录每个节点到起点的距离,以及一个前驱数组用于记录每个节点的前驱节点。在算法结束后,可以通过前驱数组来重构最短路径。 在本文中,我们…

    python 2023年5月14日
    00
合作推广
合作推广
分享本页
返回顶部