数据结构刷题(二十):17电话号码的字母组合、39组合总和、40组合总和II

news2024/10/2 22:22:29

一、电话号码的字母组合

题目链接

思路:回溯三部曲。

  • 确定回溯函数参数:题目中给的 digits,还要有一个参数就是int型的index(记录遍历第几个数字,就是用来遍历digits的,同时也代表了递归的深度),第三个参数numString(数字和字母映射)。

  • 确定终止条件:如果index 等于 输入的数字个数(digits.size)了,就return。

  • 确定单层遍历逻辑:首先要取index指向的数字,并找到对应的字符集;然后for循环来处理这个字符集

注意:

(1)解决三个问题

  • 数字和字母如何映射 (使用map或者定义一个二维数组)

  • 两个字母就两个for循环,三个字符我就三个for循环,以此类推,然后发现代码根本写不出来

  • 输入1 * #按键等等异常情况

(2)区别于普通的组合问题,本题是多个集合求组合,因为本题每一个数字代表的是不同集合,也就是求不同集合之间的组合。

(3)全局变量:一个字符串sb来收集叶子节点的结果,一个字符串数组result保存sb。

解法:

class Solution {
    // 最终结果字符数组和单次符合条件结果
    List<String> res = new ArrayList<>();
    StringBuffer sb = new StringBuffer();

    public List<String> letterCombinations(String digits) {

        if (digits.length() == 0 || digits == null)
            return res;
        String[] numString = {"", "", "abc", "def", "ghi", "jkl", "mno", "pqrs", "tuv", "wxyz"};
        backTracking(digits, numString, 0);
        return res;
    }

    public void backTracking(String digits, String[] numString ,int num) {
        if (num == digits.length()){
            res.add(sb.toString());
            return;
        }
        // digits.charAt(num)能够获取到当前的号码数字,2-9
        String t = numString[digits.charAt(num) - '0'];
        for (int i = 0; i < t.length(); i++) {
            sb.append(t.charAt(i));
            backTracking(digits, numString, num + 1);
            sb.deleteCharAt(sb.length() - 1);  //回溯
        }
    }
}

二、组合总和

题目链接

思路:基本与组合总和III类似。区别有:

  • 组合没有数量要求

  • 元素可无限重复选取

注意:

  • 如果是一个集合来求组合的话,就需要startIndex;(否则会出现重复情况)

  • 如果是多个集合取组合,各个集合之间相互不影响,那么就不用startIndex。

解法(未剪枝):

List<List<Integer>> res = new ArrayList<>();
LinkedList<Integer> path = new LinkedList<>();
public List<List<Integer>> combinationSum(int[] candidates, int target) {
    // sum,   startIndex是开始位置也是candidates的索引
    back(candidates, target, 0,0);
    return res;
}

private void back(int[] candidates, int target, int sum, int startIndex) {

    if (sum == target){
        res.add(new ArrayList<>(path));
        return;
    }
    if (sum > target)
        return;
    for (int i = startIndex; i < candidates.length; i++){
        path.add(candidates[i]);
        sum += candidates[i];
        back(candidates, target, sum, i);
        sum -= candidates[i];
        path.removeLast();
    }
}

剪枝优化:

1.对总集合排序之后,如果下一层的sum(就是本层的 sum + candidates[i])已经大于target,就可以结束本轮for循环的遍历

解法:

class Solution {
    List<List<Integer>> res = new ArrayList<>();
    LinkedList<Integer> path = new LinkedList<>();
    public List<List<Integer>> combinationSum(int[] candidates, int target) {

        Arrays.sort(candidates); // 先进行排序
        back(candidates, target, 0,  0);
        return res;
    }

    private void back(int[] candidates, int target, int sum, int startIndex) {

        if (sum == target){
            res.add(new ArrayList<>(path));
            return;
        }
        if (sum > target)
            return;
        // 多一步判断
        for (int i = startIndex; i < candidates.length && sum + candidates[i] <= target; i++){
            path.add(candidates[i]);
            sum += candidates[i];
            back(candidates, target, sum, i);
            sum -= candidates[i];
            path.removeLast();
        }
    }
}

三、组合总和II

题目链接

思路:与组合总和类似,但区别于

  1. 本题candidates 中的每个数字在每个组合中只能使用一次

  1. 本题数组candidates的元素是有重复的,而39.组合总和 (opens new window)是无重复元素的数组candidates。也就是说组合里的元素可能有重复且只使用一次,但组合之间不能重复。

  1. 本题的难点在于区别2中:集合(数组candidates)有重复元素,但还不能有重复的组合。

  1. 去重也去的是同一个树层上,重复的值。代码里就是判断i > startIndex && candidates[i] == candidates[i - 1],直接continue

解法:

class Solution {
    List<List<Integer>> res = new ArrayList<>();
    LinkedList<Integer> path = new LinkedList<>();
    public List<List<Integer>> combinationSum2(int[] candidates, int target) {
        Arrays.sort(candidates); // 先进行排序
        back(candidates, target, 0,  0);
        return res;
    }

    private void back(int[] candidates, int target, int sum, int startIndex) {

        if (sum == target){
            res.add(new ArrayList<>(path));
            return;
        }
        if (sum > target)
            return;
        for (int i = startIndex; i < candidates.length && sum + candidates[i] <= target; i++){
            // 碰到同一树层重复元素  直接continue
            if ( i > startIndex && candidates[i] == candidates[i - 1] ) {
                continue;
            }
            path.add(candidates[i]);
            sum += candidates[i];
            back(candidates, target, sum, i + 1);
            sum -= candidates[i];
            path.removeLast();
        }
    }
}

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

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

相关文章

【牛客刷题专栏】0x10:JZ8 二叉树的下一个结点(C语言编程题)

前言 个人推荐在牛客网刷题(点击可以跳转)&#xff0c;它登陆后会保存刷题记录进度&#xff0c;重新登录时写过的题目代码不会丢失。个人刷题练习系列专栏&#xff1a;个人CSDN牛客刷题专栏。 题目来自&#xff1a;牛客/题库 / 在线编程 / 剑指offer&#xff1a; 目录前言问题…

@Component实现原理

直接从关键代码开始&#xff1a; 直接找到org.springframework.context.support.AbstractApplicationContext#refresh方法&#xff0c;找到invokeBeanFactoryPostProcessors(beanFactory)方法&#xff0c;最终找org.springframework.context.support.PostProcessorRegistratio…

各种各样的锁

1.悲观锁和乐观锁 一个共享数据加了悲观锁&#xff0c;那线程每次想操作这个数据前都会假设其他线程也可能会操作这个数据&#xff0c;所以每次操作前都会上锁&#xff0c;这样其他线程想操作这个数据拿不到锁只能阻塞了。 synchronized 和 ReentrantLock是典型的悲观锁 共享…

Linux学习记录——십사 进程控制(1)

文章目录1、进程创建1、fork函数2、进程终止1、情况分类2、如何理解进程终止3、进程终止的方式3、进程等待1、进程创建 1、fork函数 fork函数从已存在进程中创建一个新进程&#xff0c;新进程为子进程&#xff0c;原进程为父进程。 #include <unistd.h> pid_t fork(vo…

论文阅读:Syntax-Aware Network for Handwritten Mathematical Expression Recognition

论文阅读&#xff1a;Syntax-Aware Network for Handwritten Mathematical Expression Recognition1 主要观点&#xff1a; 1、提出将语法信息纳入编码器-解码器网络的方法。使用一组语法规则&#xff0c;用于将每个表达式的LaTeX标记序列转换为解析树&#xff1b;用深度神经…

【vue create】一.使用vue creat搭建项目

场景&#xff1a;使用vue create脚手架快速搭建vue的项目 前提&#xff1a;需要安装node.js和cnpm以及yarn 并且cnpm需要设置为淘宝镜像&#xff0c;cnpm和yarn安装教程网上很多可以自行搜索 1.使用dos命令安装vue-cli脚手架 //这个是从镜像源下载 cnpm install -g vue/cli 查…

Google三大论文之GFS

Google三大论文之GFS Google GFS&#xff08;Google File System&#xff09; 文件系统&#xff0c;一个面向大规模数据密集型应用的、可伸缩的分布式文件系统。GFS 虽然运行在廉价的普遍硬件设备上&#xff0c;但是它依然了提供灾难冗余的能力&#xff0c;为大量客户机提供了…

接口自动化测试——多套被测环境的切换

文章目录一、意义二、实现目标三、实现方案1、使用环境管理文件2、使用不同的文件管理不同的环境&#xff08;建议使用&#xff09;3、在接口用例中指定path&#xff0c;不指定url4、环境切换a、通过环境变量进行切换b、通过命令行参数进行切换四、代码实现1、通过环境变量进行…

GPT格式的磁盘扩容

GPT格式的系统盘已经满了&#xff0c;现在需要扩充系统盘 1.怎么查看是不是GPT格式&#xff1a;fdisk -l 2.查看磁盘挂载分区情况 lsblk 2.使用parted对分区进行操作 parted /dev/sda 3.开始分区 mkpart 4.格式化sda4分区后&#xff0c;会发现分区4的文件系统已经显示为xfs…

Docker 常见操作及部署springboot、Shiro、SpringData脚手架(上)

1、查看docker 的状态 systemctl status docker 2、查看docker运行状态的详细信息 docker info 3、docker部署第一个应用 docker search nginx 拉取镜像到本地 docker pull nginx 4、查看本地的镜像信息 docker images 5、使用镜像来创建容器 docker run -d -p 1234:80 ng…

SSH配置文件解析

1.修改端口号&#xff0c;设置登录输入密码等待过期时间&#xff0c;拒绝远程登录root&#xff0c;密码为空&#xff0c;密码登录&#xff0c; [rootzzp124 ~]# vim /etc/ssh/sshd_config [rootzzp124 ~]# systemctl restart sshd.service [rootzzp124 ~]# lsof -i :2…

Superset数据探索和可视化平台入门以及案例实操

1、Superset背景 1.1、Superset概述 Apache Superset是一个现代的数据探索和可视化平台。它功能强大且十分易用&#xff0c;可对接各种数据源&#xff0c;包括很多现代的大数据分析引擎&#xff0c;拥有丰富的图表展示形式&#xff0c;并且支持自定义仪表盘。 1.2、环境说明 …

FastApi的搭建与测试

一、fastapi的安装 1-1、使用pip安装 安装fastapi的语句 pip install fastapi -i https://mirrors.aliyun.com/pypi/simple因为fastapi启动依赖于uvicorn&#xff0c;所以我们还需要安装uvicorn。 pip install uvicorn -i https://mirrors.aliyun.com/pypi/simple下面我们来…

Java学习环境一站说明(保姆级详细教学)

1.Java开发环境搭建官网下载www.oracle.com2.安装注意&#xff1a;1.选择安装位置时尽量不要安装到C盘&#xff0c;路径中不要有空格以及中文的存在2.开发人员安装的jdk中包含了jre&#xff0c;所以不需要单独安装jre3.环境变量配置打开高级系统设置2.点击环境变量3.在系统变量…

FreeSWITCH 智能呼叫流程设计

文章目录1. 智能呼叫流程2. 细节处理1. 呼叫字符串指定拨号计划2. 外呼的拨号计划3. 语音打断的支持1. 智能呼叫流程 用户与机器人对话通常都是以文本的形式进行&#xff0c;但是借助 ASR 和 TTS 技术&#xff0c;以语音电话为载体的智能呼叫系统成为可能。智能呼叫系统涉及到…

Python蓝桥杯训练:基本数据结构 [二叉树] 上

Python蓝桥杯训练&#xff1a;基本数据结构 [二叉树] 上 文章目录Python蓝桥杯训练&#xff1a;基本数据结构 [二叉树] 上一、前言二、有关二叉树理论基础1、二叉树的基本定义2、二叉树的常见类型3、二叉树的遍历方式三、有关二叉树的层序遍历的题目1、[二叉树的层序遍历](http…

网络原理之传输层协议,TCP中的主要核心机制(重点)

目录 一. 传输层中的端口号 二. UDP协议 三. TCP协议 四. TCP中的核心机制 1. 确认应答 2. 超时重传 3. 连接管理 建立连接(三次握手) 断开连接(四次挥手) 4. 滑动窗口 考虑丢包情况1&#xff1a;ack丢了 考虑丢包情况2&#xff1a;数据丢了 5. 流量控制 6. 拥塞…

学这些语言工作更吃香,Python虽然再次摘得桂冠,但在就业需求中位居第一的还得是它!

2022 IEEE 编程语言榜单发布&#xff01; IEEE Spectrum 2022 编程语言排名前十的分别是&#xff1a;Python&#xff0c;C&#xff0c;C&#xff0c;C#&#xff0c;Java&#xff0c;SQL&#xff0c;JavaScript&#xff0c;R&#xff0c;HTML&#xff0c;TypeScript。 一. Pyth…

嵌入式学习笔记——寄存器实现控制LED小灯

文章目录前言GPIO通用输出模式初始化LED小灯的GPIO原理图初始化代码初始化的效果功能函数封装直接分开宏定义两个使用条件运算符封装函数实现简单的功能前言 上一篇中&#xff0c;介绍了GPIO相关的所有寄存器&#xff0c;并在最后简单实现了一个LED灯的控制&#xff0c;由于那…

以获取笔记本电池信息为例介绍WMI的使用

注&#xff1a;本人也还没有完全弄懂WMI的原理&#xff0c;以下内容仅供参考。。。 简单来说&#xff0c;比起Win32提供的接口&#xff0c;WMI可以提供更多的系统信息&#xff0c;它本身是一个数据库架构&#xff0c;通过它可以访问、配置、管理和监视几乎所有的Windows资源&…