LeetCode 热题 HOT 100(P31~P40)

 系列文章: 

LeetCode 热题 HOT 100(P1~P10)-CSDN博客

LeetCode 热题 HOT 100(P11~P20)-CSDN博客

LeetCode 热题 HOT 100(P21~P30)-CSDN博客

LeetCode 热题 HOT 100(P31~P40)-CSDN博客

LC76minimum_window

. - 力扣(LeetCode)

题目:

给你一个字符串 s 、一个字符串 t 。返回 s 中涵盖 t 所有字符的最小子串。如果 s 中不存在涵盖 t 所有字符的子串,则返回空字符串 "" 。

输入:s = "ADOBECODEBANC", t = "ABC"
输出:"BANC"
解释:最小覆盖子串 "BANC" 包含来自字符串 t 的 'A'、'B' 和 'C'。

解法:

滑动窗口的思路,右游标可以探查到能满足条件的位置,关键的是左游标怎么确定合适的位置,因为左游标前面可能刚好是不在t中的字母。这个是第一个难点,这里通过维护一个缓存来表示字母的消耗情况,然后左游标就可以根据当前所在字母是否被过量使用判断。还有就是确定好了第一个滑动窗口left、right 之后怎么进入下一个滑动窗口的判断。这个时候left 游标要往前移动一位,同时字母消耗的缓存需要相应的处理,然后就可以继续往前滚动。

public String minWindow(String s, String t) {
        // 这里根据题目给出的限定条件,只有英文字母,所以可以使用128位asc 来表示
        int[] cache = new int[128];
        int cnt = t.length();
        for (int i = 0; i < t.length(); i++) {
            cache[t.charAt(i)]++;
        }
        int begin = 0, min = 0;
        for (int left = 0, right = 0; right < s.length(); right++) {
            if (cache[s.charAt(right)] > 0) {
                cnt--;
            }
            // 这里无差别-1,这样如果是负数就说明是t之外多余的
            cache[s.charAt(right)]--;
            if (cnt == 0) { //说明包含t
                while (cache[s.charAt(left)] < 0) { // left指针尽可能往右走
                    cache[s.charAt(left)]++;
                    left++;
                }
                final int len = right - left + 1;
                if (min == 0 || len < min) {
                    min = len;
                    begin = left;
                }

                //left 指针继续往前走,这个时候就破坏原来符合的情况了
                cache[s.charAt(left)]++;
                cnt++;
                left++;
            }
        }
        return s.substring(begin, begin + min);
    }

根据上面的思路结合代码就比较好理解了,这里缓存用int 数组表示,我发现很多时候数组比map 操作起来方便太多,这里先通过t 进行int 数组的初始化,然后在迭代过程中直接对下标-1,这样就能方便左游标是否可以往前移动。

这里还有个小技巧就是用cnt 来辅助判断是否满足条件,这样就不用轮询数组来判断。

LC78subsets

. - 力扣(LeetCode)

题目:

给你一个整数数组 nums ,数组中的元素 互不相同 。返回该数组所有可能的
子集(幂集)。解集 不能包含重复的子集。你可以按 任意顺序 返回解集。

输入:nums = [1,2,3]
输出:[[],[1],[2],[1,2],[3],[1,3],[2,3],[1,2,3]]

解法:

这类题目一闻就是递归的味道,基本套递归模板就可以了,这里要注意的是不要走回头路。

List<List<Integer>> result = new ArrayList<>();

    public List<List<Integer>> subsets(int[] nums) {
        dfs(nums, new ArrayList<>(), 0);
        return result;
    }

    private void dfs(int[] nums, List<Integer> path, int level) {
        result.add(new ArrayList<>(path));
        for (int i = level; i < nums.length; i++) {
            path.add(nums[i]);
            dfs(nums, path, i + 1);
            path.remove(path.size() - 1);
        }
    }

在上面就是通过level 控制,注意在for 循环中下一次递归的时候不是level+1 而是i+1,需要注意体会里面的区别。

LC79word_search

. - 力扣(LeetCode)

题目:

给定一个 m x n 二维字符网格 board 和一个字符串单词 word 。如果 word 存在于网格中,返回 true ;否则,返回 false 。

单词必须按照字母顺序,通过相邻的单元格内的字母构成,其中“相邻”单元格是那些水平相邻或垂直相邻的单元格。同一个单元格内的字母不允许被重复使用。

输入:board = [["A","B","C","E"],["S","F","C","S"],["A","D","E","E"]], word = "ABCCED"
输出:true

解法:

基本也是递归的思路,首选肯定能通过循环判断word 第一个字母是否匹配,这样才有继续的基础,接下来就是往上下左右遍历可能的情况,这里的难点是怎么不走回头路,主要有2种思路:

  1. 使用一个额外的缓存来标识已经走过的路。
  2. 直接在原数组中用一个特殊字符覆盖已经走过的路径,这样肯定不会走回头路,因为字符肯定不匹配。

从代码整洁性来说第二种方式更优,这里的难点是递归完下一层之后要对之前造成的影响进行复原,这个不论选择哪种方式都一样。

public boolean exist(char[][] board, String word) {
        final int m = board.length, n = board[0].length;
        for (int i = 0; i < m; i++) {
            for (int j = 0; j < n; j++) {
                if (dfs(board, i, j, word, 0)) {
                    return true;
                }
            }
        }
        return false;
    }

    private boolean dfs(char[][] board, int i, int j, String word, int index) {
        if (index == word.length()) {
            return true;
        }
        if (i < 0 || i == board.length || j < 0 || j == board[0].length || word.charAt(index) != board[i][j]) {
            return false;
        }
        // 占掉一个位置,避免走回头路
        board[i][j] = '#';
        boolean result = dfs(board, i - 1, j, word, index + 1) || dfs(board, i, j - 1, word, index + 1) ||
                         dfs(board, i + 1, j, word, index + 1) || dfs(board, i, j + 1, word, index + 1);
        board[i][j] = word.charAt(index);
        return result;
    }

LC84largest_rectangle

. - 力扣(LeetCode)

题目:

给定 n 个非负整数,用来表示柱状图中各个柱子的高度。每个柱子彼此相邻,且宽度为 1 。

求在该柱状图中,能够勾勒出来的矩形的最大面积。

输入:heights = [2,1,5,6,2,3]
输出:10
解释:最大的矩形为图中红色区域,面积为 10

解法:

一般这类n根柱子、n个容器求最大面积的问题基本是用单调递增栈来解决。但是常规的单调递增栈需要解决刚开始栈中没元素的情况,以及最后栈中还遗留元素的情况。这样代码写起来就非常冗长,而且容易出错。

这里就涉及一个单调栈的优化技巧,通过左右哨兵的加入,就可以避免处理这些异常情况。

 public int largestRectangleArea(int[] heights) {
        //维护一个单调递增栈
        Deque<Integer> stack = new ArrayDeque<>();
        stack.push(-1); //放入左哨兵,能保证他始终在栈里面
        int max = 0;
        for (int i = 0; i <= heights.length; i++) { // 这里使用heights.length 作为右哨兵
            while (getH(heights, i) < getH(heights, stack.peek())) {
                final Integer pre = stack.pop();
                int area = (i - stack.peek() - 1) * getH(heights, pre);
                max = Math.max(max, area);
            }
            stack.push(i);
        }
        return max;
    }

    private int getH(int[] heights, int index) {
        if (index == -1 || index == heights.length) {
            return -1;
        }
        return heights[index];
    }

这里通过左哨兵-1,保证了栈中肯定有值,而且下标0 的元素加入的时候天然满足递增条件。然后右哨兵-1 能保证把栈中所有元素都倒逼出来,这样就不用处理栈中还留有元素的情况。

LC94binary_tree

. - 力扣(LeetCode)

题目:

给定一个二叉树的根节点 root ,返回 它的 中序 遍历 。

解法:

如果用递归的方法是很简单的,思路也很清晰。通过代码能直观感受

 public List<Integer> inorderTraversalRecur(TreeNode root) {
        List<Integer> result = new ArrayList<>();
        if (root == null) {
            return result;
        }
        result.addAll(inorderTraversalRecur(root.left));
        result.add(root.val);
        result.addAll(inorderTraversalRecur(root.right));
        return result;
    }

如果不能使用递归,或者想要挑战下自己,那这道题的难度就上升了一个等级。核心是遍历的时候必然会用到栈,但是怎么判断当前节点是之前已经迭代过可以直接输出,还是需要迭代子节点。这里有2种解法:

第一种:颜色标记法,就是在node 中增加颜色属性,用于标识当前是否可以直接输出

public List<Integer> inorderTraversalColor(TreeNode root) {
        List<Integer> result = new ArrayList<>();
        if (root == null) {
            return result;
        }
        Deque<ColorNode> stack = new ArrayDeque<>();
        stack.push(new ColorNode(0, root));

        while (!stack.isEmpty()) {
            final ColorNode cnode = stack.pop();
            TreeNode node = cnode.node;
            if (node == null) {
                continue;
            }
            if (cnode.color == 0) {
                //需要注意这里的顺序,因为是栈所以要先放最后轮询的节点
                stack.push(new ColorNode(0, node.right));
                stack.push(new ColorNode(1, node));
                stack.push(new ColorNode(0, node.left));
            } else {
                result.add(node.val);
            }
        }
        return result;
    }

    static class ColorNode {

        /**
         * 表示是否应该输出,0,表示未遍历子树,1表示输出当前值
         */
        int color;
        TreeNode node;

        public ColorNode(int color, TreeNode node) {
            this.color = color;
            this.node = node;
        }
    }

可以发现,通过增加一个颜色字段之后这里用0、1 表示是否需要迭代子节点,整体代码逻辑就很顺,比较符合我们的逻辑。

还有一种是纯用栈的方式,这样就需要维护一个cur 指针,同时判断cur 和栈的情况,在cur 不为空的时候不断往左树迭代同时入栈,如果cur 为空就出栈,并把cur 指向右节点。

 public List<Integer> inorderTraversal(TreeNode root) {
        List<Integer> result = new ArrayList<>();
        Deque<TreeNode> stack = new ArrayDeque<>();
        TreeNode cur = root;
        while (cur != null || !stack.isEmpty()) {
            if (cur != null) {
                //不断往左子树迭代,并将当前节点保存到栈中
                stack.push(cur);
                cur = cur.left;
            } else { //左子树走到底,进行出栈并记录
                final TreeNode node = stack.pop();
                result.add(node.val);
                // 如果有右节点继续上面的过程
                cur = node.right;
            }
        }
        return result;
    }

代码虽然比上一种简洁,但是理解难度高了不少,如果吃透这种写法,基本也能玩得转二叉树类型的题目了。

LC96unique_binary

. - 力扣(LeetCode)

题目:

给你一个整数 n ,求恰由 n 个节点组成且节点值从 1 到 n 互不相同的 二叉搜索树 有多少种?返回满足题意的二叉搜索树的种数。

输入:n = 3
输出:5

解法:

看着像二叉树类型的题目,实际是动态规划的思路,核心就是动态数组的定义和动态方程的推导:

i 表示以第i个节点(从1开始)为根节点的子树,左子树个数为i,右子树个数为n-i-1
f(i) 表示i为根节点时组合个数,G(i) 表示i个节点存在的组合
G(n)=f(1)+f(2)+...+f(n)
f(i)=G(i−1)∗G(n−i)
G(n) = G(0)*G(n-1) + G(1)*G(n-2)+...+G(n-1)*G(0)
dp[n] 表示n 个节点的组合数

尝试自己推导一遍,还是能理解的,代码写出来其实很简单

 public int numTrees(int n) {
        if (n < 2) {
            return 1;
        }
        int[] dp = new int[n + 1];
        dp[0] = 1;
        dp[1] = 1;
        for (int i = 2; i <= n; i++) {
            //以第j个为 根节点时对应的组合数
            for (int j = 1; j <= i; j++) {
                dp[i] += dp[j - 1] * dp[i - j];
            }
        }
        return dp[n];
    }

但是不理解上面的推导过程,确实是看不懂。

LC98validate_binary

. - 力扣(LeetCode)

题目:

给你一个二叉树的根节点 root ,判断其是否是一个有效的二叉搜索树。

有效 二叉搜索树定义如下:

  • 节点的左

    子树

    只包含 小于 当前节点的数。
  • 节点的右子树只包含 大于 当前节点的数。
  • 所有左子树和右子树自身必须也是二叉搜索树。

解法:

刚一开始,就想到了用递归的方法来解决这个问题,先判断当前节点是否合法(左值<当前值<右值),递归判断左节点,最后递归右节点。结果啪啪打脸,这里存在一个问题是这样的判断不严谨,可能左子树其中一个叶子节点满足直属父节点和兄弟节点的关系,但是并不满足跟祖父节点的约束。

因此这里需要基于中序遍历是递增关系这条更严格的约束条件。

 Integer preVal;

    public boolean isValidBST(TreeNode root) {
        if (root == null) {
            return true;
        }
        if (!isValidBST(root.left)) {
            return false;
        }
        if (preVal != null && root.val <= preVal) {
            return false;
        }
        preVal = root.val;
        return isValidBST(root.right);
    }

注意这里使用了包装类型Integer,方便处理第一次赋值的特殊情况。

LC101symmetric_tree

. - 力扣(LeetCode)

题目:

给你一个二叉树的根节点 root , 检查它是否轴对称。

解法:

这类一想就有很多子情况需要判断的题目,基本要么递归要么动态规划。或者说他俩基本是一回事。

实际就是判断左节点跟右节点是否对称

 public boolean isSymmetric(TreeNode root) {
        if (root == null) {
            return true;
        }
        return dfs(root.left, root.right);
    }

    private boolean dfs(TreeNode left, TreeNode right) {
        if (left == null && right == null) {
            return true;
        }
        // 到这里说明left和right 不会同时为null,那么其中有一个null 就说明不对称
        if (left == null || right == null) {
            return false;
        }
        if (left.val != right.val) {
            return false;
        }
        return dfs(left.left, right.right) && dfs(left.right, right.left);
    }

可以看到复杂的地方是对边界情况的处理,是否都为null啊,或者其中一个为null,当都不为null 的时候判断是否相等。然后最后再递归,注意这里left.left 对应right.right,这个看下轴对称示意图就很好理解了。

LC102binary_tree

. - 力扣(LeetCode)

题目:

给你二叉树的根节点 root ,返回其节点值的 层序遍历 。 (即逐层地,从左到右访问所有节点)。

输入:root = [3,9,20,null,null,15,7]
输出:[[3],[9,20],[15,7]]

解法:

首先想到的是通过队列进行广度遍历,但是第二个问题是怎么知道当前是那一层,因为需要按层输出。这里有个小技巧就是在遍历队列的时候,直接判断当前队列的长度,这个就是当前层的节点数。

 public List<List<Integer>> levelOrder(TreeNode root) {
        List<List<Integer>> result = new ArrayList<>();
        if (root == null) {
            return result;
        }
        Queue<TreeNode> queue = new ArrayDeque<>();
        queue.add(root);
        while (!queue.isEmpty()) {
            final int size = queue.size();
            List<Integer> levelResult = new ArrayList<>();
            for (int i = 0; i < size; i++) {
                final TreeNode node = queue.poll();
                levelResult.add(node.val);
                if (node.left != null) {
                    queue.offer(node.left);
                }
                if (node.right != null) {
                    queue.offer(node.right);
                }
            }
            result.add(levelResult);
        }
        return result;
    }

行数虽多,但是很多是对边界情况的处理,整体还是很好理解的。

LC104maximum_depth

. - 力扣(LeetCode)

题目:

给定一个二叉树 root ,返回其最大深度。

二叉树的 最大深度 是指从根节点到最远叶子节点的最长路径上的节点数。

输入:root = [3,9,20,null,null,15,7]
输出:3

题解:

这个也是递归的思路,从左子树、右子树中选择最深的节点数+1 就是当前的最大深度。

 public int maxDepth(TreeNode root) {
        if (root == null) {
            return 0;
        }
        return Math.max(maxDepth(root.left), maxDepth(root.right)) + 1;
    }

代码非常简单,只能说递归真是非常强大。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

设计一个通知系统

设计的系统支持不同类型消息的发送&#xff0c;例如push消息&#xff0c;sms消息和邮箱消息&#xff0c;能够支持千万级别的发送&#xff0c;保证消息推送的幂等性。系统结构图如下&#xff1a; 系统架构图中各组件作用说明&#xff1a; device/setting/user info&#xff1a;…

反向迭代器的底层

文章目录 1.迭代器分类2.迭代器使用3.模拟实现迭代器3.1 各个类的迭代器3.2 所有容器的迭代器(迭代器适配器) 1.迭代器分类 迭代器按照定义方式分成以下四种。 正向迭代器&#xff0c;定义方法如下&#xff1a; 容器类名::iterator 迭代器名; 常量正向迭代器&#xff0c;定…

Swagger转换成Excel文件

1、添加swagger解析依赖包&#xff1a; <dependency><groupId>io.swagger.parser.v3</groupId><artifactId>swagger-parser</artifactId><version>2.1.12</version></dependency>2、示例代码&#xff1a; package com.rlclou…

第十四届蓝桥杯题解:平方差 ,更小的数,买瓜,网络稳定性(货车运输)

目录 平方差 更小的数 买瓜 网络稳定性&#xff08;货车运输&#xff09; 货车运输 平方差 这道题就是数论的题&#xff0c;不难想到一个数m可以拆成(a-b)(ab)&#xff0c;其实(a-b)和(ab)就是m的一对因子&#xff0c;不妨设为x和y。 则有&#xff1a; abx; a-by; x*ym; 联…

《由浅入深学习SAP财务》:第2章 总账模块 - 2.7 总账模块报表 -2.7.1 对外报表:资产负债表及利润表

总账模块报表既包括对外报告的资产负债表、损益表、现金流量表&#xff0c;也包括企业自身用于查询和分析的各类报表&#xff0c;如科目余额表等。 2.7.1 对外报表&#xff1a;资产负债表及利润表 在SAP中&#xff0c;出具资产负债表和利润表的标准方法是先在后台建立一套“会…

大模型化身数据魔法师,降低NLP高置信误判

关注公众号【AI论文解读】回复: 论文解读 获取本文论文 引言&#xff1a;NLP模型的高置信错误与脆弱性问题 在自然语言处理&#xff08;NLP&#xff09;领域&#xff0c;模型的预测性能优化往往伴随着高置信错误&#xff08;high confidence errors&#xff09;的产生&#x…

【MATLAB源码-第49期】基于蚁群算法(ACO)算法的栅格路径规划,输出最佳路径图和算法收敛曲线图。

操作环境&#xff1a; MATLAB 2022a 1、算法描述 蚁群算法是一种模拟自然界蚂蚁觅食行为的启发式优化算法。在蚁群系统中&#xff0c;通过模拟蚂蚁之间通过信息素沟通的方式来寻找最短路径。 在栅格路径规划中&#xff0c;蚁群算法的基本步骤如下&#xff1a; 1. 初始化: …

LeetCode-热题100:104. 二叉树的最大深度

题目描述 给定一个二叉树 root &#xff0c;返回其最大深度。 二叉树的 最大深度 是指从根节点到最远叶子节点的最长路径上的节点数。 示例 1&#xff1a; 输入&#xff1a; root [3,9,20,null,null,15,7] 输出&#xff1a; 3 示例 2&#xff1a; 输入&#xff1a; root …

直驱式风电机组的发电机和双馈风电机组的发电机发电机generator的区别

直驱式风电机组的发电机和双馈风电机组的发电机在结构和工作原理上有明显的区别&#xff1a; 直驱式风电机组的发电机&#xff1a; 结构简单&#xff0c;通常由永磁同步发电机构成。直接将风轮的转动与发电机的转子连接&#xff0c;无需传动系统。没有齿轮箱&#xff0c;因此减…

GPT图解:大模型是怎样构建的,书籍PDF分享

今天又来给大家推荐一本大模型方面的书籍<GPT图解:大模型是怎样构建的>本书将以生动活泼的笔触&#xff0c;将枯燥的技术细节化作轻松幽默的故事和缤纷多彩的图画&#xff0c;引领读者穿梭于不同技术的时空&#xff0c;见证自然语言处理技术的传承、演进与蜕变。 在这本…

求存款本息和(C语言)

一、运行结果&#xff1b; 二、源代码&#xff1b; # define _CRT_SECURE_NO_WARNINGS # include <stdio.h> # include <math.h>int main() {//初始化变量值&#xff1b;double P 1000, r1 0.015, r2 0.021, r3 0.0275, r4 0.03, r5 0.0035;int judge 0;//…

【C语言】——字符串函数的使用与模拟实现(上)

【C语言】——字符串函数 前言一、 s t r l e n strlen strlen 函数1.1、函数功能1.2、函数的使用1.3、函数的模拟实现&#xff08;1&#xff09;计数法&#xff08;2&#xff09;递归法&#xff08;3&#xff09;指针 - 指针 二、 s t r c p y strcpy strcpy 函数2.1、函数功能…

Go语言开发工具Vscode配置

Go语言开发工具Vscode配置方法分享&#xff1a; 1.下载安装vscode https://code.visualstudio.com/ 2.汉化vscode 3.vscode中安装Go语言插件 源自&#xff1a;大地老师Golang语言beego入门实战视频教程下载地址

5、LMDeploy 量化部署 LLMVLM实战(homework)

基础作业&#xff08;结营必做&#xff09; 完成以下任务&#xff0c;并将实现过程记录截图&#xff1a; 配置lmdeploy运行环境 由于环境依赖项存在torch&#xff0c;下载过程可能比较缓慢。InternStudio上提供了快速创建conda环境的方法。打开命令行终端&#xff0c;创建一…

项目实现:Boost搜索引擎

一.项目背景 当前已经有许多上市公司做了搜索引擎&#xff0c;比如说百度&#xff0c;搜狗&#xff0c;360等等&#xff0c;这些项目都是很大的项目&#xff0c;有很高的技术门槛&#xff0c;我们自己实现一个完整的搜索引擎是不可能的&#xff0c;但是我们可以写一个简单的搜…

【ARM 裸机】硬件平台简介

硬件平台采用的是正点原子的 I.MX6ULL-MINI 开发板&#xff0c;分为底板和核心板&#xff1b; 1、底板 正点原子 Mini 开发板的外形尺寸为 100mm*130mm&#xff0c;I.MX6U-Mini 开发板底板板载资源如下&#xff1a; ◆ 1 个核心板接口&#xff0c;支持 I.MX6ULL 核心板。 ◆ 1…

梯度提升树(Gradient Boosting Trees)

通过5个条件判定一件事情是否会发生&#xff0c;5个条件对这件事情是否发生的影响力不同&#xff0c;计算每个条件对这件事情发生的影响力多大&#xff0c;写一个梯度提升树&#xff08;Gradient Boosting Trees&#xff09;模型程序,最后打印5个条件分别的影响力。 示例一 梯…

RobotFramework功能自动化测试框架基础篇

概念 RobotFramework是什么&#xff1f; Robot Framework是一款python编写的功能自动化测试框架。具备良好的可扩展性&#xff0c;支持关键字驱动&#xff0c;可以同时测试多种类型的客户端或者接口&#xff0c;可以进行分布式测试执行。主要用于轮次很多的验收测试和验收测试…

力扣练习题(2024/4/14)

1接雨水 给定 n 个非负整数表示每个宽度为 1 的柱子的高度图&#xff0c;计算按此排列的柱子&#xff0c;下雨之后能接多少雨水。 示例 1&#xff1a; 输入&#xff1a;height [0,1,0,2,1,0,1,3,2,1,2,1] 输出&#xff1a;6 解释&#xff1a;上面是由数组 [0,1,0,2,1,0,1,3,2…

基于 net/http 抽象出 go 服务优雅停止的一般思路

和其他语言相比&#xff0c;Go 中有相同也有不同&#xff0c;相同的是实现思路上和其他语言没啥差异&#xff0c;不同在于 Go 采用的是 goroutine channel 的并发模型&#xff0c;与传统的进程线程相比&#xff0c;实现细节上存在差异。 本文将从实际场景和它的一般实现方式展…