首页 > 编程语言 >leetcode刷题day22|回溯算法Part01( 77. 组合 、216. 组合总和 III、17.电话号码的字母组合)

leetcode刷题day22|回溯算法Part01( 77. 组合 、216. 组合总和 III、17.电话号码的字母组合)

时间:2024-09-20 20:49:14浏览次数:17  
标签:216 遍历 return 组合 int result 回溯 字母组合 path

前言:回溯是递归的副产品,只要有递归就会有回溯,回溯函数也就是递归函数。回溯是暴力穷举解法,效率并不高。但一些问题只能使用回溯来解决。

回溯法,一般可以解决如下几种问题:

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

回溯三部曲

1、回溯函数返回值以及参数:
在回溯算法中,返回值一般为void,参数按需定义。
2、回溯函数终止条件
3、回溯搜索的遍历过程:回溯法一般是在集合中递归搜索,集合的大小构成了树的宽度,递归的深度构成的树的深度。
回溯函数遍历过程伪代码如下:

for (选择:本层集合中元素(树中节点孩子的数量就是集合的大小)) {
    处理节点;
    backtracking(路径,选择列表); // 递归
    回溯,撤销处理结果
}

for循环就是遍历集合区间,可以理解一个节点有多少个孩子,这个for循环就执行多少次。

backtracking这里自己调用自己,实现递归。

for循环可以理解是横向遍历,backtracking(递归)就是纵向遍历,这样就把这棵树全遍历完了,一般来说,搜索叶子节点就是找的其中一个结果了。

77. 组合

思路:暴力解法就是直接套两层循环,但如果k值很大循环写起来会很麻烦,所以选择使用回溯,本质上是在树中加for循环。

回溯三部曲:
1、递归函数的返回值以及参数:定义两个全局变量,一个用来存放符合条件单一结果path,一个用来存放符合条件结果的集合result。传入参数:集合元素个数n和取数个数k,另外需要一个变量startIndex,记录本层递归中集合从哪里开始遍历。
2、终止条件:path数组的大小如果达到k,说明我们找到了一个子集大小为k的组合了,说明到达叶子节点了,加入结果集合。
3、单层搜索的过程:for循环每次从startIndex开始遍历,然后用path保存取到的节点i。

代码如下:

class Solution {
    List<List<Integer>> result;
    List<Integer> path;
    public List<List<Integer>> combine(int n, int k) {
        result=new ArrayList<>();
        path=new LinkedList<>();
        backTracking(n,k,1);
        return result;
    }
    public void backTracking(int n,int k,int startIndex){
        if(path.size()==k){
            result.add(new ArrayList(path));
            return;
        }
        for(int i=startIndex;i<=n;i++){
            path.add(i);
            backTracking(n,k,i+1);
            path.removeLast();
        }
    }
}

注意:这里可以通过剪枝进行优化,如果是从四个元素中取四个元素,那么每一层中的后面元素都不用进入for循环了,即可以修改for循环为

for (int i = startIndex; i <= n - (k - path.size()) + 1; i++) // i为本次搜索的起始位置

其中,已经选择的元素个数:path.size();
还需要的元素个数为: k - path.size();
在集合n中至多要从该起始位置 : n - (k - path.size()) + 1,开始遍历

216. 组合总和 III

思路:这个题目和前一个差不多,直接回溯三部曲。

回溯三部曲:
1、递归函数的返回值以及参数:定义两个全局变量,一个用来存放符合条件单一结果path,一个用来存放符合条件结果的集合result。传入参数:相加之和n和取数个数k,另外需要一个变量startIndex,记录本层递归中集合从哪里开始遍历。
2、终止条件:path数组的大小如果达到k且n=0,加入结果集合;如果n不等于0,直接return。
3、单层搜索的过程:for循环每次从startIndex开始遍历,然后用path保存取到的节点i,n减去当前值传入下一次递归调用。

代码如下:

class Solution {
    List<List<Integer>> result;
    List<Integer> path;
    public List<List<Integer>> combinationSum3(int k, int n) {
        result=new ArrayList<>();
        path=new LinkedList<>();
        backTracking(n,k,1);
        return result;
    }
    public void backTracking(int n,int k,int startIndex){
        if(path.size()==k && n==0){
            result.add(new ArrayList(path));
            return;
        }else if(path.size()==k && n==0) return;
        for(int i=startIndex;i<=9;i++){
            path.add(i);
            backTracking(n-i,k,i+1);
            path.removeLast();
        }
    }
}

17.电话号码的字母组合

思路:这个题与前面题目的思路的区别是for循环里面是1-9;本题2-9每个数字对应的集合均不同。

要解决如下三个问题:

1、数字和字母如何映射:直接用数组的下标和字符串来对应
2、如何进行回溯?

回溯三部曲:
1、确定回溯函数参数:全局变量:字符串s来收集叶子节点的结果,一个字符串数组result保存结果。参数:string digits,index记录遍历第几个数字了。
2、终止条件:如果s的长度等于数字字符串的长度,存入resul,return
3、单层遍历逻辑:取index指向的数字,并找到对应的字符集,for循环处理字符集,此时的for每次都从0开始,不是原来不断变化的数字下标了。

代码如下:

class Solution {
    List<String> result;
    StringBuilder s;
    public List<String> letterCombinations(String digits) {
        result=new ArrayList<>();
        s=new StringBuilder();
        if(digits==null || digits.length()==0){
            return result;
        }
        String[] words={"","","abc","def","ghi","jkl","mno","pqrs","tuv","wxyz"};
        backTracing(digits,words,0);
        return result;
    }
    public void backTracing(String digits,String[] words,int index){
        if(index==digits.length()){
            result.add(s.toString());
            return;
        }
        String word=words[digits.charAt(index)-'0'];
        for(int i=0;i<word.length();i++){
            s.append(word.charAt(i));
            backTracing(digits,words,index+1);
            s.deleteCharAt(s.length()-1);
        }
    }
}

标签:216,遍历,return,组合,int,result,回溯,字母组合,path
From: https://blog.csdn.net/m0_51007517/article/details/142331676

相关文章

  • leetcode刷题day23|回溯算法Part02(39. 组合总和 、40.组合总和II、131.分割回文串)
    39.组合总和思路:这个题与77.组合的差异在于元素可以无限制重复被选取,那么只需要更改startIndex即可,每一层递归都可以从头选用元素。回溯三部曲与77.组合基本一致。代码如下:classSolution{List<List<Integer>>result=newArrayList<>();List<Integer>pa......
  • 代码随想录算法训练营第十六天 | Javascript | 力扣Leetcode | 回溯 | 77. 组合、216.
    目录前言简介题目链接:77.组合题目链接:216.组合总和3题目链接:17.电话号码的字母组合前言踏平坎坷成大道,斗罢艰险又出发!自律的尽头是自控,自控的尽头是硬控。愿道友们披荆斩棘,终能得偿所愿。简介本人是小几年经验的前端开发,算法基础只有力扣几十道题,非常薄......
  • 深入解析Vue 3组合函数:提高代码复用性和模块化的最佳实践
    随着Vue3的引入,组合式API(CompositionAPI)带来了更灵活的代码组织方式,组合函数作为其核心部分,能够显著提升代码的可维护性、复用性和模块化。在这篇文章中,我们将通过一个具体的表格管理和分页功能的示例,详细介绍如何使用组合函数来构建更加高效和清晰的Vue3应用。1.组合函数......
  • 基于Vue 3组合函数的分页、搜索与排序实践 —— nbsaas-boot项目的实际应用
    随着前端框架的不断发展,Vue3引入的组合式API(CompositionAPI)为开发者提供了一种更灵活、更强大的逻辑复用方式。组合函数(CompositionFunction)可以将通用逻辑抽取成独立模块,便于在不同组件间共享和复用。本文将基于nbsaas-boot项目,详细介绍如何通过Vue3的组合函数实现分页、......
  • 代码随想录算法训练营,9月19日 | 39. 组合总和,40.组合总和II,131.分割回文串
    39.组合总和题目链接:39.组合总和文档讲解︰代码随想录(programmercarl.com)视频讲解︰组合总和日期:2024-09-19想法:组合总和类型题,允许重复使用元素,递归不+1就行。Java代码如下:classSolution{List<Integer>path=newArrayList<>();List<List<Integer>>res=n......
  • 分公司=一部门——组合模式
    文章目录分公司=一部门——组合模式分公司不就是一部门吗?组合模式透明方式与安全方式何时使用组合模式公司管理系统组合模式好处分公司=一部门——组合模式分公司不就是一部门吗?时间:5月10日19点地点:小菜、大鸟住所的客厅人物:小菜、大鸟“大鸟,请教你一个问......
  • Day 19 回溯法part01| LeetCode 77.组合,216. 组合总和 III,17. 电话号码的字母组合
    理论基础回溯法(回溯搜索法)回溯函数就是递归函数本质是穷举解决的问题组合问题(不强调元素顺序,需去重)切割问题子集问题:一个N个数的集合里有多少符合条件的子集排列问题(强调元素顺序)棋盘问题:N皇后回溯法模板(可抽象为树形结构——N叉树来解决问题)递归返回值以及......
  • P2163 [SHOI2007] 园丁的烦恼
    中学的时候年轻气盛,应该拿主席树把这道题艹过去了。正好复习离线二维数点,再做了一遍。我们把询问差分安排到x轴上,然后y轴用树状数组统计即可。注意到此题要离散化,但是询问中的x可能不在原序列中。不过这不要紧,记得二分完减一就好。constintN=5e5+5,S=1e7+5;intn,m......