代码随想录二刷回溯算法-组合问题总结

news2025/1/22 21:55:27

回溯算法实际上也是一种暴力算法,利用树型结构的回溯与剪枝从而解决问题 

解题步骤主要分三步:1.确立回溯函数的参数 2.确立终止条件 3.确立单层遍历逻辑


组合问题 

77. 组合

这道题目就是经典的组合问题

如果我们使用for循环来进行暴力求解,那么当n和k越来越多,我们嵌套的for循环也就会越来越多

所以我们使用回溯算法来解决,我们先画出一个树状图

那么解决回溯算法的核心就在于使用for循环来进行横向遍历使用递归回溯来完成纵向遍历

画图举例

使用递归完成纵向遍历

使用for循环完成横向遍历

1.确立回溯函数的参数:

要传入数组,为了让函数知道下一次遍历应该从哪开始,所以也需要传入起始坐标

public void backtracking(int[] nums,int startIndex)

2.确立终止条件:当我们的path的长度到达2时就将path添加到result里面去

3.确立单层遍历逻辑: 设置for循环使得数组横向遍历

for(int i = startIndex;i < nums.length;i++){
    path.add(nums[i]);
    backtracing(nums,i+1);
    path.removeLast();
}

 最终代码如下

class Solution {
    List<List<Integer>> result= new ArrayList<>();
    LinkedList<Integer> path = new LinkedList<>();
    public List<List<Integer>> combine(int n, int k) {
        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();
        }
    }
}

216. 组合总和 III

这题的解题思路实际上和上一题是一样的,虽然题目中写道:每个数字 最多使用一次,但实际上我们使用横向遍历的时候每一次遍历的数字都是不同的,所以我们本题在上一题的基础上只需要改变 2.确立终止条件 即可

 2.确立终止条件:

		if (path.size() == k) {
			if (sum == targetSum) result.add(new ArrayList<>(path));
			return;
		}

其余思路不变,具体代码如下:

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

    public List<List<Integer>> combinationSum3(int k, int n) {
        combinationSum3Helper(k,n,1);
        return result;
    }

    public void combinationSum3Helper(int k,int n,int startIndex){
        if(path.size() == k){
            int sum = 0;

            for(int i : path){
                sum += i;
            }

            if(sum == n){
                result.add(new ArrayList(path));
            }

            return;
        }

        for(int i = startIndex;i <= 9;i++){
            path.add(i);
            combinationSum3Helper(k,n,i+1);
            path.removeLast();
        }
    }   
}

17. 电话号码的字母组合

首先画出本题的树状图

1.确立回溯函数的参数:

首先要知道每个按键所对应的字母组合所以要传入一个String数组;

当我们选择了 2 这个按键,我们还需要知道下一个按键的位置,所以要传入按键位置以及题目所给定的数字组合;

当然我们还需要下标来确认遍历的位置在哪所以也需要startIndex,但是本题的startIndex实际上就是按键位置,所以代码如下:

public void letterCombinationsHelper(String digits,String[] numsString,int num)

2.确立终止条件

当我们的按键数字与我们的数字组合长度相同时那么说明此时就到叶子节点需要进行返回了

所以代码如下:

        if(num == digits.length()){
            result.add(stringBuffer.toString());
            return;
        }

3.确立单层遍历逻辑

我们首先要有一个String来接收我们的字母组合,然后再利用for循环的横向遍历来手机不同的字母组合

所以代码如下:

        String str = numsString[digits.charAt(num) - '0'];
        
        for(int i = 0;i < str.length();i++){
            stringBuffer.append(str.charAt(i));
            letterCombinationsHelper(digits,numsString,num + 1);
            stringBuffer.deleteCharAt(stringBuffer.length() - 1);
        }

最后完整代码如下:

class Solution {
    List<String> result = new ArrayList();
    
    public List<String> letterCombinations(String digits) {
        if(digits == null || digits.length() == 0){
            return result;
        }
        String[] numsString = {" "," ","abc","def","ghi","jkl","mno","pqrs","tuv","wxzy"};
        letterCombinationsHelper(digits,numsString,0);
        return result;
    }

    StringBuffer stringBuffer = new StringBuffer();

    public void letterCombinationsHelper(String digits,String[] numsString,int num){
        if(num == digits.length()){
            result.add(stringBuffer.toString());
            return;
        }
        
        String str = numsString[digits.charAt(num) - '0'];
        
        for(int i = 0;i < str.length();i++){
            stringBuffer.append(str.charAt(i));
            letterCombinationsHelper(digits,numsString,num + 1);
            stringBuffer.deleteCharAt(stringBuffer.length() - 1);
        }
    }
}

组合总和问题 

39. 组合总和

这题的解题思路实际上还是用到回溯算法,但是我们在纵向遍历的时候需要时注意的是传递到下一层递归的起始位置与传递的起始位置相同,原因在于题目中要求的是不同的组合,也就是说加入我们从3开始进行纵向遍历,那么下一层递归的起始位置也是3

首先画出树状图: 

1.确立回溯函数的参数:首先要传递数组以及题目要求的目标值,然后传递起始位置,然后还需要知道每一层的总和是多少(这里传不传其实都可以)

public void combinationSumHelper(int[] candidates,int sum,int target,int index)

2.确立终止条件:当总和达到目标值时就添加到ans中并且返回

        if(sum == target){
            result.add(new ArrayList(path));
            return;
        }

3.确立单层遍历逻辑:

我们在确定单层遍历逻辑的时候需要进行剪枝操作陷入死循环,也就是说,当总和以及超过目标值时就应该直接返回

        for(int i = index;i < candidates.length;i++){
            if(sum + candidates[i] > target){
                break;
            }
            sum += candidates[i];
            path.add(candidates[i]);
            combinationSumHelper(candidates,sum,target,i);
            path.removeLast();
            sum -= candidates[i];
        }

最后完整代码如下

class Solution {
    List<List<Integer>> result = new ArrayList();
    LinkedList<Integer> path = new LinkedList();
    int sum = 0;
    public List<List<Integer>> combinationSum(int[] candidates, int target) {
        if(candidates.length == 0){
            return result;
        }
        Arrays.sort(candidates);
        combinationSumHelper(candidates,target,0);
        return result;
    }

    public void combinationSumHelper(int[] candidates,int target,int index){
        if(sum == target){
            result.add(new ArrayList(path));
            return;
        }

        for(int i = index;i < candidates.length;i++){
            if(sum + candidates[i] > target){
                break;
            }
            sum += candidates[i];
            path.add(candidates[i]);
            combinationSumHelper(candidates,target,i);
            path.removeLast();
            sum -= candidates[i];
        }
    }
}

40. 组合总和 II

这题实际上和上面那题大体思路上是一样的,但是这题的要求是每个组合中不能出现重复的数字

这里的重复的数字实际上指的是数组中重复的下标所对应的数字

这也解释了为什么能出现 [1,1,6] 这种组合

那么为了解决不出现重复的下标所对应的数字,我们使用startIndex来标记每次开始的下标位置哪,从而避免重复出现重复的下标所对应的数字

同时还要求解集不能包含重复的组合,那么我们可以将数组进行排序,当遇到相同的数字时就跳过知道遇到不同的数字,这样就可以不包含重复的组合,因为当两个数字相同的时候,那么他们两个对应的组合一定会有重复的。

1.确立回溯函数的参数:

这题我们只需要知道每层递归的起始位置在哪即可

public void combinationSum2Helper(int[] candidates,int target,int index)

2.确立终止条件

当我们的总和达到了目标值,那么就将结果添加到ans中并且返回

        if(sum == target){
            result.add(new ArrayList(path));
        }

3.确立单层遍历逻辑

 同样的,为了避免死循环,当总和超过目标值就直接跳出循环直接返回了

if(sum + candidates[i] > target){
    break;
}

当我们在单层遍历的过程中需要确定该下标是否被使用过

      if ( i > start && candidates[i] == candidates[i - 1] ) {
        continue;
      }

 这里为什么 i 是 i > start 而不是 i > 0 呢?如果是 i > 0的话那么就一定会省略掉 [1,1,6] 这种组合

当我们设置成  i > start 以后既能够保证在同一层里面不出现重复的组合又能保证不会省略掉像[1,1,6] 这种组合

    for ( int i = start; i < candidates.length; i++ ) {
      if(sum + candidates[i] <= target){
          break;
      }

      if (i > start && candidates[i] == candidates[i - 1] ) {
        continue;
      }

      sum += candidates[i];
      path.add( candidates[i] );
      // i+1 代表当前组内元素只选取一次
      backTracking( candidates, target, i + 1 );
      sum -= candidates[i];
      path.removeLast();
    }

全部代码如下:

class Solution {
    List<List<Integer>> result = new ArrayList();
    LinkedList<Integer> path = new LinkedList();
    int sum = 0;

    public List<List<Integer>> combinationSum2(int[] candidates, int target) {
        use = new boolean[candidates.length];
        Arrays.fill(use,false);
        Arrays.sort(candidates);
        combinationSum2Helper(candidates,target,0);
        return result;
    }

    public void combinationSum2Helper(int[] candidates,int target,int index){
        if(sum == target){
            result.add(new ArrayList(path));
        }
        for(int i = index;i < candidates.length;i++){
            if(sum + candidates[i] > target){
                break;
            }
            if(i > 0 && candidates[i] == candidates[i-1] && !use[i-1]){
                continue;
            }
            sum += candidates[i];
            path.add(candidates[i]);
            combinationSum2Helper(candidates,target,i+1);
            sum -= candidates[i];
            path.removeLast();
        }
    }
}

组合分割问题

组合分割问题的核心在于分割,我们通常使用 startIndex 下标来作为分割线,从而达到分割的效果,通常这类题目的解法分为三步,

1.使用for循环完成横向遍历 使用回溯完成纵向遍历(基本的回溯操作)

2.在每一层内使用函数判断分割后的字符串是否合法

3.达到终止条件以后添加到结果内

131. 分割回文串

这道题的核心问题在于如何分割,我们使用startIndex来完成分割,通过告诉下一层的起始位置在哪从而完成分割字符串

首先我们画出树状图

 

可以看到我们通过不停的使用startIndex就可以完成对字符串的分割,将字符串分割成不同的子串,再将子串放进判断是否合法的函数内进行判断,如果合法就放进收集结果的链表内,如果 不合法那么就将移动分割线到 下一位

        for(int i = startIndex;i < s.length();i++){
            if(isPartition(s,startIndex,i)){
                String str = s.substring(startIndex,i+1);
                dq.addLast(str);
            }else{
                continue;
            }
            partitionHelper(s,i+1);
            dq.removeLast();
        }

对于是否合法的函数需要根据题意来编写代码

    public boolean isPartition(String s,int startIndex,int endIndex){
        int left = startIndex;
        int right = endIndex;
        while(left <= right){
            if(s.charAt(left) != s.charAt(right)){
                return false;
            }
            left++;
            right--;
        }
        return true;
    }

当我们的分割线超过了字符串的长度时,此时就已经分割完毕了所以我们要将收集到的结果放进结果集内

        if(startIndex >= s.length()){
            ans.add(new ArrayList(dq));
            return;
        }

最后贴上完整代码

class Solution {
    List<List<String>> ans = new ArrayList();
    Deque<String> dq = new LinkedList();

    public List<List<String>> partition(String s) {
        partitionHelper(s,0);
        return ans;
    }

    public void partitionHelper(String s,int startIndex){
        if(startIndex >= s.length()){
            ans.add(new ArrayList(dq));
            return;
        }

        for(int i = startIndex;i < s.length();i++){
            if(isPartition(s,startIndex,i)){
                String str = s.substring(startIndex,i+1);
                dq.addLast(str);
            }else{
                continue;
            }
            partitionHelper(s,i+1);
            dq.removeLast();
        }
    }

    public boolean isPartition(String s,int startIndex,int endIndex){
        int left = startIndex;
        int right = endIndex;
        while(left <= right){
            if(s.charAt(left) != s.charAt(right)){
                return false;
            }
            left++;
            right--;
        }
        return true;
    }
}

93. 复原 IP 地址

这道题目实际上还是分割问题,我们根据刚才的思路将树状图画出来(关键的分支)

本题的关键在于合法IP地址的判断函数,题目要求 每个整数位于 0 到 255 之间组成,且不能含有前导 0,当前导为0时要求该整数只能为0

所以我们合法IP地址判断应该这么写

    public boolean isVaild(StringBuffer stringBuffer,int start,int end){
        String s = stringBuffer.toString(); 
        
        if(start > end){
            return false;
        }

        char[] str = s.toCharArray();

        if(start != end && str[start] == '0'){
            return false;
        }

        int num = 0;

        for(int i = start;i <= end;i++){
            if(str[i] > '9' || str[i] < '0'){
                return false;
            }
            int number = str[i] - '0';
            
            num = num * 10 + number;

            if(num > 255){
                return false;
            }
        }

        return true;
    }

我们的回溯函数里面for循环的代码编写同之前的解法是一样的,首先判断该子串是否合法,如果合法就进入到下一层递归进行分割,如果不合法那么就直接跳出递归,回溯到上一层

        for(int i = startIndex;i < str.length();i++){
            if(isVaild(str,startIndex,i)){
                str.insert(i+1,".");
                restoreIpAddressesHelper(str,i+2,pointNums+1);
                str.deleteCharAt(i+1);
            }else{
                break;
            }
        }

我们的终止条件是当字符串中出现了三个分割点时就添加到结果集中,这里要注意的是由于我们判断子串是否合法的时候判断的是分割点前面的子串而不是分割点后面的子串所以很有可能会出现下面这种情况

所以为了防止这种情况的发生,我们需要在终止条件的地方再次进行判断

        if(pointNums == 3){
            if(isVaild(str,startIndex,str.length()-1)){
                ans.add(str.toString());
            }
            return;
        }

 总体代码如下

class Solution {
    List<String> ans = new ArrayList();
    public List<String> restoreIpAddresses(String s) {
        StringBuffer str = new StringBuffer(s);
        restoreIpAddressesHelper(str,0,0);
        return ans;
    }

    public void restoreIpAddressesHelper(StringBuffer str,int startIndex,int pointNums){
        //如果有三个点那么就直接返回了
        if(pointNums == 3){
            if(isVaild(str,startIndex,str.length()-1)){
                ans.add(str.toString());
            }
            return;
        }

        for(int i = startIndex;i < str.length();i++){
            if(isVaild(str,startIndex,i)){
                str.insert(i+1,".");
                restoreIpAddressesHelper(str,i+2,pointNums+1);
                str.deleteCharAt(i+1);
            }else{
                break;
            }
        }
    }

    public boolean isVaild(StringBuffer stringBuffer,int start,int end){
        String s = stringBuffer.toString(); 
        
        if(start > end){
            return false;
        }

        char[] str = s.toCharArray();

        if(start = end && str[start] == '0'){
            return false;
        }

        int num = 0;

        for(int i = start;i <= end;i++){
            if(str[i] > '9' || str[i] < '0'){
                return false;
            }
            int number = str[i] - '0';
            
            num = num * 10 + number;

            if(num > 255){
                return false;
            }
        }

        return true;
    }
}

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

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

相关文章

spring boot-Resolved element must not contain multiple elements 警告

首先强调一下&#xff0c;此问题不影响程序运行。 报错信息&#xff1a; package org.springframework.util; ...public abstract class Assert ...public static void state(boolean expression, String message) {if (!expression) {throw new IllegalStateException(messa…

融合康养产业、乐享宜居灞桥,西安市灞桥康养论坛即将举办

随着我国人口老龄化进程的不断加速以及人们的健康意识不断提高&#xff0c;我国康养产业逐步发展壮大。9月15日&#xff0c;以“融合康养产业、乐享宜居灞桥”为主题的灞桥康养论坛将在西安市灞桥区盛大召开。 据悉&#xff0c;此次论坛由西安市人民政府、陕西省民政厅主办&am…

2024浙大MEM提面拿优秀笔试如何冲刺备考

浙大工程师学院对于参加浙大提前批面试并获得优秀资格的考生&#xff0c;提供了一个“笔试达到联考国家线即可拟录取”的优惠政策。这确实是吸引很多MEM考生参加提前批面试的原因之一。但是&#xff0c;即使获得了优秀资格&#xff0c;考生仍然需要在后续的联考笔试中达到一定的…

JAVASE 窗口按钮

本文目录 1、前言2、JFrame、JButton3、JLabl4、ImageIcon 1、前言 java提供了很多已经写好了的类供我们使用&#xff0c;而我们没必要去细腻研究它的构成原理&#xff0c;就好比我们让我们编程让机器人动起来&#xff0c;没必要细腻研究机器人每个器件是怎么做出来的一样&…

免杀对抗-ShellCode上线+回调编译执行+混淆变异算法

C/C --ShellCode-免杀对抗 介绍&#xff1a; shellcode是一段用于利用软件漏洞而执行的代码&#xff0c;shellcode为16进制的机器码&#xff0c;因为经常让攻击者获得shell而得名。我们经常在CS里面生成指定编程语言的payload&#xff0c;而这个payload里面就是一段十六进制的机…

Centos7 完全断网离线环境下安装MySQL 8.0.33 图文教程

Centos7 完全断网离线环境安装MySQL 8.0.33 图文教程 1.1前言1.2 下载离线安装包1.3 将下载好的离线安装包上传到Centos 7 服务器1.3.1 方式一:联网环境下可利用rz命令进行文件上传1.3.2 方式二:断网环境下使用 XFtp 等软件工具进行上传1.4 解压安装包1.5 执行安装脚本1.6 重…

《极客时间:数据结构与算法之美》【数据结构与算法】

本篇博客是学习过程中的笔记整理和个人思考。原文链接&#xff1a;https://time.geekbang.org/column/intro/100017301 开篇词 | 从今天起&#xff0c;跨过“数据结构与算法”这道坎01 | 为什么要学习数据结构和算法&#xff1f;02 | 如何抓住重点&#xff0c;系统高效地学习数…

CTF入门学习笔记——Crypto密码(古典密码)

文章目录 CTF入门学习笔记——Crypto密码&#xff08;古典密码&#xff09;凯撒密码看我回旋踢 摩斯密码摩斯 维吉尼亚密码Vigenre 栅栏密码篱笆墙的影子 栅栏密码篱笆墙的影子 猪圈密码待补充 CTF入门学习笔记——Crypto密码&#xff08;古典密码&#xff09; &#x1f680;&a…

领域驱动设计:DDD重构中台业务模型

文章目录 如何避免重复造轮子&#xff1f;如何构建中台业务模型&#xff1f; 如何避免重复造轮子&#xff1f; 要避免重复建设&#xff0c;就要理解中台的理念和思想。“中台是企业级能力复用平台”&#xff0c;“复用”用白话说就是重复使用&#xff0c;就是要避免重复造轮子…

深入解析OLED透明屏的工作原理与优势,智能家居的未来之选

OLED透明屏作为一项突破性的显示技术&#xff0c;不仅具备出色的视觉效果&#xff0c;还带来了全新的功能和应用。 在这篇文章中&#xff0c;尼伽将深入探讨OLED透明屏的功能特点&#xff0c;介绍其在各个领域的广泛应用&#xff0c;并提供实用的案例和数据&#xff0c;希望看…

认识 Express

1. 初识 Express 1.1 Express 简介 1. 什么是 Express 官方给出的概念&#xff1a;Express 是基于 Node.js 平台&#xff0c;快速、开放、极简的 Web 开发框架。 通俗的理解&#xff1a;Express 的作用和 Node.js 内置的 http 模块类似&#xff0c;是专门用来创建 Web …

“批量随机字母命名文件,轻松管理你的文件库“

你是否曾经遇到过文件命名混乱&#xff0c;难以管理的问题&#xff1f;为了解决这个问题&#xff0c;我们推出了一款全新的文件改名工具&#xff0c;它可以帮助你批量给文件名添加一个随机字母&#xff0c;让你的文件库更加有序、易于管理。 首先第一步&#xff0c;我们要进入…

Python 交易指南:利用 RSI

一、说明 RSI是相对强弱指数&#xff08;Relative Strength Index&#xff09;的缩写&#xff0c;是一种技术指标。该指标是用来测量股票或其他交易品种的价格波动强度和速度的&#xff0c;属于动量型指标。RSI常用于技术分析和交易策略中&#xff0c;可以帮助交易者判断市场的…

JAVA 的四种访问权限

在Java编程中&#xff0c;访问权限是非常重要的概念&#xff0c;因为它可以保证代码的安全性和封装性。访问权限有四种&#xff0c;分别是public、protected、default和private。 private&#xff1a;如果一个类的方法或者变量被private修饰&#xff0c;那么这个类的方法或者变…

程序执行的四个阶段

程序执行的四个阶段 对于一段helloc.c的程序 #include <stdio.h>int main() {printf("hello, world\n");return 0; }为了在系统上运行程序&#xff0c;每条C语句都必须被其他程序转化为一系列的低级机器语言指令。然后这些指令按照一种称为可执行目标程序的格…

网络层--IP协议

引入&#xff1a; IP协议主要解决什么问题呢&#xff1f; IP协议提供一种将数据从主机&#xff21; 发送到 主机&#xff22;的能力。&#xff08;有能力不一定能做到&#xff0c;比如小明很聪明&#xff0c;可以考100分&#xff0c;但是他也不是每次搜能考100分&#xff0…

持安科技入选数说安全《2023中国网络安全市场年度报告》

近日&#xff0c;网络安全产业研究平台数说安全发布《2023中国网络安全市场年度报告》&#xff0c;报告共分为158页核心报告&#xff0c;及番外篇《网安融资新星及融资过亿企业介绍》&#xff0c;作为以甲方身份创业的零信任办公安全明星企业&#xff0c;持安科技以网安融资新星…

【leetcode 力扣刷题】重复叠加字符串匹配

重复叠加字符串匹配 686. 重复叠加字符串匹配 686. 重复叠加字符串匹配 题目链接&#xff1a;686. 重复叠加字符串匹配 题目内容&#xff1a; 理解题意&#xff0c;可以发现题目还是要求我们做字符串匹配。只是查询串不是简单的a&#xff0c;而是a的叠加&#xff0c;并且这个…

国外LEAD收款渠道介绍:Wise收款教程

在国内做国外的Affiliate marketing&#xff0c;收款还是有些麻烦的。以前用Payoneer挺方便&#xff0c;包括clickbank&#xff0c;amazon等联盟都挺顺利的回款&#xff0c;不过自从Digitalstore24的一笔联盟款发送之后&#xff0c;没有到账&#xff0c;然后就收到款项需要审核…

windows下安装redis扩展库

1.根据PHP版本号&#xff0c;编译器版本号和CPU架构 选择php_redis和php_igbinary文件(如果是选择线程的情况下需要再去配置php5ts.dll) windows.php.net - /downloads/pecl/releases/redis/ windows.php.net - /downloads/pecl/releases/igbinary/ php_igbinary-3.1.2-7.2-…