系列文章: 

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> result = new ArrayList<>();

public List> subsets(int[] nums) {

dfs(nums, new ArrayList<>(), 0);

return result;

}

private void dfs(int[] nums, List 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种思路:

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

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

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 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 inorderTraversalRecur(TreeNode root) {

List 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 inorderTraversalColor(TreeNode root) {

List result = new ArrayList<>();

if (root == null) {

return result;

}

Deque 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 inorderTraversal(TreeNode root) {

List result = new ArrayList<>();

Deque 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> levelOrder(TreeNode root) {

List> result = new ArrayList<>();

if (root == null) {

return result;

}

Queue queue = new ArrayDeque<>();

queue.add(root);

while (!queue.isEmpty()) {

final int size = queue.size();

List 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;

}

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

文章来源

评论可见,请评论后查看内容,谢谢!!!评论后请刷新页面。