Skip to content

Commit 304e409

Browse files
Update
1 parent 48be019 commit 304e409

10 files changed

+189
-42
lines changed

problems/0098.验证二叉搜索树.md

Lines changed: 35 additions & 34 deletions
Original file line numberDiff line numberDiff line change
@@ -2,63 +2,64 @@
22

33
## 思路
44

5+
中序遍历下,输出的二叉搜索树节点的数值是有序序列,有了这个特性,**验证二叉搜索树,就相当于变成了判断一个序列是不是递增的了。**
6+
7+
所以代码实现上,我们就使用递归法来中序遍历,遍历的过程中判断节点上的数值是不是递增的就可以了。
8+
59
这道题目比较容易陷入两个陷阱:
10+
611
* 陷阱1 :[10,5,15,null,null,6,20] 这个case 要考虑道
7-
* 陷阱2:样例中根节点的val 可能是-2147483648
12+
![二叉搜索树](https://img-blog.csdnimg.cn/20200812191501419.png)
813

9-
中序遍历之后 输出的顺序就应该是一个从大到小的顺序, 可以使用一个全局变量进行比较。
14+
10的右子树只能包含大于当前节点的数,而右面出现了一个6 这就不符合了!
1015

11-
## C++代码
16+
* 陷阱2:样例中根节点的val 可能是int的最小值
17+
18+
问题可以进一步演进:如果样例中根节点的val 可能是longlong的最小值 又要怎么办呢?看下文解答!
1219

20+
## C++代码
1321

14-
[10,5,15,null,null,6,20] 为什么预期结果是 false.... 这是经典陷阱
15-
错误代码
22+
定于全局变量初始化为long long最小值
1623
```
1724
class Solution {
1825
public:
26+
long long maxVal = LONG_MIN;
1927
bool isValidBST(TreeNode* root) {
2028
if (root == NULL) return true;
21-
22-
if (root->left != NULL && root->right != NULL) {
23-
if (root->val > root->left->val && root->val < root->right->val) {
24-
return true;
25-
} else {
26-
return false;
27-
}
28-
}
29-
30-
if (root->left != NULL && root->right == NULL) {
31-
if (root->val > root->left->val) {
32-
return true;
33-
}else {
34-
return false;
35-
}
36-
}
37-
if (root->left == NULL && root->right != NULL) {
38-
if (root->val < root->right->val) {
39-
return true;
40-
}else {
41-
return false;
42-
}
43-
}
44-
if (root->left == NULL && root->right == NULL) return true;
45-
return isValidBST(root->left) && isValidBST(root->right);
29+
bool left = isValidBST(root->left);
30+
// 中序遍历,验证遍历的元素是不是从小到大
31+
if (maxVal < root->val) maxVal = root->val;
32+
else return false;
33+
bool right = isValidBST(root->right);
34+
return left && right;
4635
}
4736
};
4837
```
4938

50-
正确代码
39+
其实因为后台数据有int最小值测试用例,所以都改成了longlong最小值。
40+
41+
如果测试数据中有 longlong的最小值,怎么办?不可能在初始化一个更小的值了吧。 建议避免 初始化最小值,如下方法取到最左面的数值:
42+
5143
```
5244
class Solution {
5345
public:
54-
long long val = LONG_MIN;
46+
long long maxVal = 0; // 记录中序遍历的过程中出现过的最大值
47+
bool flag = false; // 标记是否取到了最左面节点的数值
5548
bool isValidBST(TreeNode* root) {
5649
if (root == NULL) return true;
5750
bool left = isValidBST(root->left);
58-
if (val < root->val) val = root->val;// 中序遍历,这里相当于从大到小进行比较
59-
else return false;
51+
if (!flag) {
52+
maxVal = root->val;
53+
flag = true;
54+
} else {
55+
// 中序遍历,这里相当于从大到小进行比较
56+
if (maxVal < root->val) maxVal = root->val;
57+
else return false;
58+
}
6059
bool right = isValidBST(root->right);
6160
return left && right;
6261
}
6362
};
6463
```
64+
65+
> 更多算法干货文章持续更新,可以微信搜索「代码随想录」第一时间围观,关注后,回复「Java」「C++」 「python」「简历模板」「数据结构与算法」等等,就可以获得我多年整理的学习资料。

problems/0144.二叉树的前序遍历.md

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -177,7 +177,7 @@ public:
177177
,然后在反转result数组,输出的结果顺序就是左右中了,如下图:
178178
![前序到后序](https://img-blog.csdnimg.cn/20200808200338924.png)
179179

180-
所以后序遍历只需要前序遍历的代码稍作修改就可以了,代码如下:
180+
**所以后序遍历只需要前序遍历的代码稍作修改就可以了,代码如下:**
181181

182182
```
183183
class Solution {

problems/0199.二叉树的右视图.md

Lines changed: 15 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,21 @@ https://leetcode-cn.com/problems/binary-tree-right-side-view/
33

44
## 思路
55

6-
广度优先搜索模板题目,层序遍历的时候,将每一层的最后元素放入result数组中
6+
这里再讲一遍二叉树的广度优先遍历(层序遍历)
7+
8+
需要借用一个辅助数据结构队列来实现,**队列先进先出,符合一层一层遍历的逻辑,而是用栈先进后出适合模拟深度优先遍历也就是递归的逻辑。**
9+
10+
使用队列实现广度优先遍历,动画如下:
11+
12+
<video src='../video/102二叉树的层序遍历.mp4' controls='controls' width='640' height='320' autoplay='autoplay'> Your browser does not support the video tag.</video></div>
13+
14+
这样就实现了层序从左到右遍历二叉树。
15+
16+
建议先做一下这道题目[0102.二叉树的层序遍历](https://github.com/youngyangyang04/leetcode/blob/master/problems/0102.二叉树的层序遍历.md), 在做这道,就会发现这是一道 广度优先搜索模板题目,层序遍历的时候,将每一层的最后元素放入result数组中,
17+
18+
层序遍历的时候,将单层的最后面的元素放进数组中,随后返回数组就可以了。
19+
20+
代码如下:
721

822
## C++代码
923

problems/0209.长度最小的子数组.md

Lines changed: 9 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,15 @@
22
https://leetcode-cn.com/problems/minimum-size-subarray-sum/
33

44
## 思路
5-
这道题目 暴力解法当然是 两个for循环,然后不断的寻找符合条件的子序列
6-
这块我们还可以使用滑动窗口的细想来做这道题。所谓滑动窗口,**就是不断的调节子序列的起始位置,从而得出我们要想的结果**
5+
这道题目暴力解法当然是 两个for循环,然后不断的寻找符合条件的子序列,时间复杂度很明显是O(n^2) 。
6+
7+
还可以使用滑动窗口的细想来做这道题。所谓滑动窗口,**就是不断的调节子序列的起始位置,从而得出我们要想的结果**
8+
9+
这里还是以题目中的示例来举例,s=7, 数组是 2,3,1,2,4,3,来看一下动画效果:
10+
11+
<video src='../video/209.长度最小的子数组.mp4' controls='controls' width='640' height='320' autoplay='autoplay'> Your browser does not support the video tag.</video></div>
12+
13+
代码如下:
714

815
## 暴力解法
916

problems/0239.滑动窗口最大值.md

Lines changed: 45 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -7,12 +7,46 @@ https://leetcode-cn.com/problems/sliding-window-maximum/
77

88
暴力方法,遍历一遍的过程中每次从窗口中在找到最大的数值,这样很明显是O(n * k)的算法。
99

10-
有的同学可能会想用一个大顶堆也就是优先级队列来存放这个窗口里的k个数字,这样就可以知道最大的最大值是多少了, 但是问题是这个窗口是移动的,而大顶堆每次只能弹出最大值,我们无法移除其他数值,这样就造成大顶堆维护的不是滑动窗口里面的数值了。所以不能用大顶堆。
10+
有的同学可能会想用一个大顶堆也就是优先级队列来存放这个窗口里的k个数字,这样就可以知道最大的最大值是多少了, **但是问题是这个窗口是移动的,而大顶堆每次只能弹出最大值,我们无法移除其他数值,这样就造成大顶堆维护的不是滑动窗口里面的数值了。所以不能用大顶堆。**
1111

12-
使用单调队列,即单调递减或单调递增的队列。 它不是一个独立的数据结构,需要使用其他数据结构来实现单调队列,例如: deque,deque是双向队列,可以选择 从头部或者尾部pop,同样也可以从头部或者尾部push。
12+
**使用单调队列,即单调递减或单调递增的队列。它不是一个独立的数据结构,需要使用其他数据结构来实现单调队列**,例如: deque,deque是双向队列,可以选择 从头部或者尾部pop,同样也可以从头部或者尾部push。
1313

1414
不要以为实现的单调队列就是 对窗口里面的数进行排序,如果排序的话,那和优先级队列又有什么区别了呢。
1515

16+
使用deque实现的单调队列如下:(代码详细注释)
17+
18+
```
19+
class MyQueue { //单调队列(从大到小)
20+
public:
21+
deque<int> que; // 使用deque来实现单调队列
22+
// 每次弹出的时候,比较当前要弹出的数值是否等于队列前端的数值,如果相等在pop数据,当然也要判断队列当前是否为空。
23+
void pop(int value) {
24+
if (!que.empty() && value == que.front()) {
25+
que.pop_front();
26+
}
27+
}
28+
// 如果push的数值大于后端的数值,那么就将队列后端的数值弹出,直到push的数值小于等于 队列后端的数值为止。
29+
// 然后再将数值push到队列后端,这样就保持了队列里的数值是单调从大到小的了。
30+
void push(int value) {
31+
while (!que.empty() && value > que.back()) {
32+
que.pop_back();
33+
}
34+
que.push_back(value);
35+
36+
}
37+
// 查询当前队列里的最大值 直接返回队列前端也就是front就可以了。
38+
int front() {
39+
return que.front();
40+
}
41+
};
42+
```
43+
44+
动画解释如下:
45+
<video src='../video/0239.滑动窗口最大值.mp4' controls='controls' width='640' height='320' autoplay='autoplay'> Your browser does not support the video tag.</video></div>
46+
47+
48+
这样我们就用deque实现了一个单调队列,接下来解决滑动窗口最大值的问题就很简单了。
49+
1650
详情看代码吧,已经简洁。
1751

1852
## C++代码
@@ -55,5 +89,14 @@ public:
5589
}
5690
};
5791
```
92+
来看一下时间复杂度,时间复杂度: O(n),
93+
有的同学可能想了,在队里中 push元素的过程中,还有pop操作呢,感觉不是纯粹了O(n)。
94+
95+
其实,大家可以自己观察一下单调队列的实现,nums 中的每个元素最多也就被 push_back 和 pop_back 一次,没有任何多余操作,所以整体的复杂度还是 O(n)。
96+
97+
空间复杂度因为我们定义一个辅助队列,所以是O(k)。
98+
99+
100+
58101
> 更过算法干货文章持续更新,可以微信搜索「代码随想录」第一时间围观,关注后,回复「Java」「C++」 「python」「简历模板」「数据结构与算法」等等,就可以获得我多年整理的学习资料。
59102

problems/0700.二叉搜索树中的搜索.md

Lines changed: 77 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,83 @@ https://leetcode-cn.com/problems/search-in-a-binary-search-tree/
33

44
## 思路
55

6-
注意这里是二叉搜索树,是已经排序的了,两种实现,递归和迭代
6+
### 递归法
7+
8+
先来看递归的实现方式。
9+
10+
依然从递归三要素开始分析:
11+
12+
* 确定递归函数的参数和返回值
13+
* 确定终止条件
14+
* 确定单层递归的逻辑
15+
16+
1. 确定递归函数的参数和返回值
17+
18+
递归函数的参数传入的就是根节点和要搜索的数值,返回的就是以这个搜索数值所在的节点。
19+
20+
代码如下:
21+
22+
```
23+
TreeNode* searchBST(TreeNode* root, int val)
24+
```
25+
26+
2. 确定终止条件
27+
28+
如果root为空,或者找到这个数值了,就返回root节点。
29+
30+
```
31+
if (root == NULL || root->val == val) return root;
32+
```
33+
34+
3. 确定单层递归的逻辑
35+
36+
来看一下二叉搜索树的单层递归逻辑有何不同, 因为二叉搜索树的节点是有序的,所以可以有方向的去搜索,如果root->val > val,搜索左子树,如果root->val < val,就搜索右子树,最后如果都没有搜索到,就返回NULL。
37+
38+
代码如下:
39+
40+
```
41+
if (root->val > val) return searchBST(root->left, val);
42+
if (root->val < val) return searchBST(root->right, val);
43+
return NULL;
44+
```
45+
46+
这里可能会疑惑,在递归遍历的时候,什么时候直接return 递归函数的返回值,什么时候不用加这个 return, 这取决于对递归函数的定义,这里定义的递归函数,就是返回 要查找的元素所在的节点,而这个节点就是我们所求,所以直接return递归函数的返回值。
47+
48+
整体代码如下:
49+
50+
```
51+
TreeNode* searchBST(TreeNode* root, int val) {
52+
if (root == NULL || root->val == val) return root;
53+
if (root->val > val) return searchBST(root->left, val);
54+
if (root->val < val) return searchBST(root->right, val);
55+
return NULL;
56+
}
57+
```
58+
59+
### 迭代法
60+
61+
一提到二叉树遍历的迭代法,可能立刻想起使用栈来模拟深度遍历,使用队列来模拟广度遍历,其实因为二叉搜索树的特殊性,也就是节点的有序性,可以不使用辅助栈或者队列就可以写出迭代法。
62+
63+
对于一般二叉树,模拟递归的过程中还有一个回溯的过程,例如走一个左方向的分支走到头了,那么要调头,在走右分支。而对于二叉搜索树,不需要回溯的过程,因为节点的有序性就帮我们确定了搜索的方向。
64+
65+
看如下图中的这颗二叉搜索树,例如要搜索元素为3的节点,我们不需要搜索其他节点,也不需要做回溯,查找的路径已经规划好了。
66+
![二叉搜索树](https://img-blog.csdnimg.cn/20200812190213280.png)
67+
68+
迭代法代码如下:
69+
70+
```
71+
class Solution {
72+
public:
73+
TreeNode* searchBST(TreeNode* root, int val) {
74+
while (root != NULL) {
75+
if (root->val > val) root = root->left;
76+
else if (root->val < val) root = root->right;
77+
else return root;
78+
}
79+
return NULL;
80+
}
81+
};
82+
```
783

884
## C++代码
985

problems/1047.删除字符串中的所有相邻重复项.md

Lines changed: 7 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,13 @@ https://leetcode-cn.com/problems/remove-all-adjacent-duplicates-in-string/
33

44
## 思路
55

6-
这道题目就像是我们玩过的游戏对对碰, 可以把字符串放到与一个栈中,然后如果相同的话 栈就弹出,这样最后栈里剩下的元素都是相邻不相同的元素了
6+
这道题目就像是我们玩过的游戏对对碰, 可以把字符串放到与一个栈中,然后如果相同的话 栈就弹出,这样最后栈里剩下的元素都是相邻不相同的元素了。
7+
8+
如动画所示:
9+
10+
<video src='../video/1047.删除字符串中的所有相邻重复项.mp4' controls='controls' width='640' height='320' autoplay='autoplay'> Your browser does not support the video tag.</video></div>
11+
12+
从栈中弹出剩余元素,此时是字符串ac,因为从栈里弹出的元素是倒叙的,所以在对字符串进行反转一下,就得到了最终的结果。
713

814
## C++代码
915

video/0239.滑动窗口最大值.mp4

1.82 MB
Binary file not shown.
663 KB
Binary file not shown.
1.7 MB
Binary file not shown.

0 commit comments

Comments
 (0)