一、哈希
1. 两数之和
-
题面:
给定一个整数数组
nums
和一个整数目标值target
,请你在该数组中找出 和为目标值target
的那 两个 整数,并返回它们的数组下标。你可以假设每种输入只会对应一个答案,并且你不能使用两次相同的元素。
你可以按任意顺序返回答案。
- 只会存在一个有效答案
-
示例 1:
1
2
3输入:nums = [2,7,11,15], target = 9
输出:[0,1]
解释:因为 nums[0] + nums[1] == 9 ,返回 [0, 1] 。示例 2:
1
2输入:nums = [3,2,4], target = 6
输出:[1,2]示例 3:
1
2输入:nums = [3,3], target = 6
输出:[0,1]
暴力枚举肯定是可以的,但二重循环,时间复杂度高。
1 | class Solution { |
时间复杂度:.
空间复杂度:.
哈希表
1 | class Solution { |
时间复杂度:.
空间复杂度:.
2. 字母异位词分组
-
题面:
给你一个字符串数组,请你将 字母异位词 组合在一起。可以按任意顺序返回结果列表。
字母异位词 是由重新排列源单词的所有字母得到的一个新单词。
strs[i]
仅包含小写字母
-
示例 1:
1
2输入: strs = ["eat", "tea", "tan", "ate", "nat", "bat"]
输出: [["bat"],["nat","tan"],["ate","eat","tea"]]示例 2:
1
2输入: strs = [""]
输出: [[""]]示例 3:
1
2输入: strs = ["a"]
输出: [["a"]]
字符串自身排序作为键
1 | class Solution { |
时间复杂度:.
空间复杂度:.
还可以计数 26 个字母出现的次数,成为一个数组,这个数组作为键。
写的太复杂,这里不给代码了。
3. 最长连续序列
-
题面:
给定一个未排序的整数数组
nums
,找出数字连续的最长序列(不要求序列元素在原数组中连续)的长度。请你设计并实现时间复杂度为 的算法解决此问题。
-
示例 1:
1
2
3输入:nums = [100,4,200,1,3,2]
输出:4
解释:最长数字连续序列是 [1, 2, 3, 4]。它的长度为 4。示例 2:
1
2输入:nums = [0,3,7,2,5,8,4,6,0,1]
输出:9
注意点:
-
使用
unordered_set
-
不要重复查找,有前序数字的数字肯定不会是最长序列的开端。
例如,有4,5,6,7,那么,5 肯定不会是最长序列的开端,同理,6,7也不会是。
怎么判断:看是不是存在前序数字即可,比如 5 看是不是存在 4。
1 | class Solution { |
时间复杂度:.
空间复杂度:.
二、双指针
4. 移动零
-
题面:
给定一个数组
nums
,编写一个函数将所有 0 移动到数组的末尾,同时保持非零元素的相对顺序。请注意 ,必须在不复制数组的情况下原地对数组进行操作。
-
示例 1:
1
2输入: nums = [0,1,0,3,12]
输出: [1,3,12,0,0]示例 2:
1
2输入: nums = [0]
输出: [0]
双指针法
1 | class Solution { |
时间复杂度:.
空间复杂度:.
5. 盛最多水的容器
-
题面:
给定一个长度为
n
的整数数组height
。有n
条垂线,第i
条线的两个端点是(i, 0)
和(i, height[i])
。找出其中的两条线,使得它们与 x 轴共同构成的容器可以容纳最多的水。
返回容器可以储存的最大水量。
说明:你不能倾斜容器。
-
示例 1:
1
2
3输入:[1,8,6,2,5,4,8,3,7]
输出:49
解释:图中垂直线代表输入数组 [1,8,6,2,5,4,8,3,7]。在此情况下,容器能够容纳水(表示为蓝色部分)的最大值为 49。示例 2:
1
2输入:height = [1,1]
输出:1
关键在于思考怎么移动指针。
1 | class Solution { |
时间复杂度:.
空间复杂度:.
6. 三数之和
-
题面:
给你一个整数数组
nums
,判断是否存在三元组[nums[i], nums[j], nums[k]]
满足i != j
、i != k
且j != k
,同时还满足nums[i] + nums[j] + nums[k] == 0
。请你返回所有和为 0 且不重复的三元组。注意:答案中不可以包含重复的三元组。
-
示例 1:
1
2
3
4
5
6
7
8输入:nums = [-1,0,1,2,-1,-4]
输出:[[-1,-1,2],[-1,0,1]]
解释:
nums[0] + nums[1] + nums[2] = (-1) + 0 + 1 = 0 。
nums[1] + nums[2] + nums[4] = 0 + 1 + (-1) = 0 。
nums[0] + nums[3] + nums[4] = (-1) + 2 + (-1) = 0 。
不同的三元组是 [-1,0,1] 和 [-1,-1,2] 。
注意,输出的顺序和三元组的顺序并不重要。示例 2:
1
2
3输入:nums = [0,1,1]
输出:[]
解释:唯一可能的三元组和不为 0 。示例 3:
1
2
3输入:nums = [0,0,0]
输出:[[0,0,0]]
解释:唯一可能的三元组和为 0 。
三重循环查找不可避免。
- 查找时,同一个位置不能出现相同的数字,因此先排序是必要的。
- 可以发现,要
a+b+c = 0
那么,确定a
的情况下,b
和c
是对立的,此消彼长,因此不需要完完全全的三重循环,二重循环即可。 - 注意
c
下标应该大于b
下标。
1 | class Solution { |
时间复杂度:.
空间复杂度:. 主要是排序的消耗。
7. 接雨水
-
题面:
给定
n
个非负整数表示每个宽度为1
的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水。 -
示例 1:
1
2
3输入:height = [0,1,0,2,1,0,1,3,2,1,2,1]
输出:6
解释:上面是由数组 [0,1,0,2,1,0,1,3,2,1,2,1] 表示的高度图,在这种情况下,可以接 6 个单位的雨水(蓝色部分表示雨水)。示例 2:
1
2输入:height = [4,2,0,3,2,5]
输出:9
动态规划
1 | class Solution { |
时间复杂度:.
空间复杂度:.
双指针,可以降低空间复杂度
1 | class Solution { |
时间复杂度:.
空间复杂度:.
三、滑动窗口
8. 无重复字符的最长子串
-
题面:
给定一个字符串 s ,请你找出其中不含有重复字符的 最长 子串 的长度。
子串:是字符串中连续的 非空 字符序列。
-
示例 1:
1
2
3输入: s = "abcabcbb"
输出: 3
解释: 因为无重复字符的最长子串是 "abc",所以其长度为 3。示例 2:
1
2
3输入: s = "bbbbb"
输出: 1
解释: 因为无重复字符的最长子串是 "b",所以其长度为 1。示例 3:
1
2
3
4输入: s = "pwwkew"
输出: 3
解释: 因为无重复字符的最长子串是 "wke",所以其长度为 3。
请注意,你的答案必须是 子串 的长度,"pwke" 是一个子序列,不是子串。
1 | class Solution { |
9. 找到字符串中所有字母异位词
-
题面:
给定两个字符串
s
和p
,找到s
中所有p
的 异位词 的子串,返回这些子串的起始索引。不考虑答案输出的顺序。
-
示例 1:
1
2
3
4
5输入: s = "cbaebabacd", p = "abc"
输出: [0,6]
解释:
起始索引等于 0 的子串是 "cba", 它是 "abc" 的异位词。
起始索引等于 6 的子串是 "bac", 它是 "abc" 的异位词。示例 2:
1
2
3
4
5
6输入: s = "abab", p = "ab"
输出: [0,1,2]
解释:
起始索引等于 0 的子串是 "ab", 它是 "ab" 的异位词。
起始索引等于 1 的子串是 "ba", 它是 "ab" 的异位词。
起始索引等于 2 的子串是 "ab", 它是 "ab" 的异位词。
1 | class Solution { |
四、子串
10. 和为 K 的子数组
-
题面:
给你一个整数数组
nums
和一个整数k
,请你统计并返回 该数组中和为k
的子数组的个数 。子数组是数组中元素的连续非空序列。
-
示例 1:
1
2输入:nums = [1,1,1], k = 2
输出:2示例 2:
1
2输入:nums = [1,2,3], k = 3
输出:2
首先,朴素做法:
不佳,会报出:超出内存限制。
1 | class Solution { |
时间复杂度:.
空间复杂度:.
改进变量使用:
1 | class Solution { |
时间复杂度:.
空间复杂度:.
只求前缀和,并且用哈希表存储:
1 | class Solution { |
时间复杂度:.
空间复杂度:.
11.滑动窗口最大值
-
题面:
给你一个整数数组
nums
,有一个大小为k
的滑动窗口从数组的最左侧移动到数组的最右侧。你只可以看到在滑动窗口内的k
个数字。滑动窗口每次只向右移动一位。返回 滑动窗口中的最大值 。
-
示例 1:
1
2
3
4
5
6
7
8
9
10
11输入:nums = [1,3,-1,-3,5,3,6,7], k = 3
输出:[3,3,5,5,6,7]
解释:
滑动窗口的位置 最大值
--------------- -----
[1 3 -1] -3 5 3 6 7 3
1 [3 -1 -3] 5 3 6 7 3
1 3 [-1 -3 5] 3 6 7 5
1 3 -1 [-3 5 3] 6 7 5
1 3 -1 -3 [5 3 6] 7 6
1 3 -1 -3 5 [3 6 7] 7示例 2:
1
2输入:nums = [1], k = 1
输出:[1]
朴素做法,超出时间限制:
1 | class Solution { |
时间复杂度:.
空间复杂度:.
优先队列是最应该想到的正确方法:
1 | class Solution { |
时间复杂度:.
空间复杂度:.
优化算法,可以只用单调队列(要用双端队列,队头队尾都能插入删除,但人为保持单调特性)
1 | class Solution { |
时间复杂度:.
空间复杂度:.
12. 最小覆盖子串
-
题面:
给你一个字符串
s
、一个字符串t
。返回s
中涵盖t
所有字符的最小子串。如果s
中不存在涵盖t
所有字符的子串,则返回空字符串""
。注意:
- 对于
t
中重复字符,我们寻找的子字符串中该字符数量必须不少于t
中该字符数量。 - 如果
s
中存在这样的子串,我们保证它是唯一的答案。
- 对于
-
示例 1:
1
2
3输入:s = "ADOBECODEBANC", t = "ABC"
输出:"BANC"
解释:最小覆盖子串 "BANC" 包含来自字符串 t 的 'A'、'B' 和 'C'。示例 2:
1
2
3输入:s = "a", t = "a"
输出:"a"
解释:整个字符串 s 是最小覆盖子串。示例 3:
1
2
3
4输入: s = "a", t = "aa"
输出: ""
解释: t 中两个字符 'a' 均应包含在 s 的子串中,
因此没有符合条件的子字符串,返回空字符串。
- 哈希表的数量检测不可避免
- 滑动窗口,右指针
r
扩大,直到check true
。再左指针l
缩小,直到check false
。
1 | class Solution { |
时间复杂度:.
空间复杂度:.
五、普通数组
13. 最大子数组和
-
题面:
给你一个整数数组 nums ,请你找出一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和。
子数组 是数组中的一个连续部分。
**进阶:**如果你已经实现复杂度为
O(n)
的解法,尝试使用更为精妙的 分治法 求解。 -
示例 1:
1
2
3输入:nums = [-2,1,-3,4,-1,2,1,-5,4]
输出:6
解释:连续子数组 [4,-1,2,1] 的和最大,为 6 。示例 2:
1
2输入:nums = [1]
输出:1示例 3:
1
2输入:nums = [5,4,-1,7,8]
输出:23
滑动窗口,自己想的
1 | class Solution { |
时间复杂度:.
空间复杂度:.
不用记录那么多,动态规划即可
1 | class Solution { |
时间复杂度:.
空间复杂度:.
14. 合并区间
-
题面:
以数组
intervals
表示若干个区间的集合,其中单个区间为 。请你合并所有重叠的区间,并返回 一个不重叠的区间数组,该数组需恰好覆盖输入中的所有区间 。 -
示例 1:
1
2
3输入:intervals = [[1,3],[2,6],[8,10],[15,18]]
输出:[[1,6],[8,10],[15,18]]
解释:区间 [1,3] 和 [2,6] 重叠, 将它们合并为 [1,6].示例 2:
1
2
3输入:intervals = [[1,4],[4,5]]
输出:[[1,5]]
解释:区间 [1,4] 和 [4,5] 可被视为重叠区间。
时间安排问题,都是先按照开始时间排序再合并。
1 | class Solution { |
时间复杂度:.
空间复杂度:.
15. 轮转数组
-
题面:
给定一个整数数组
nums
,将数组中的元素向右轮转k
个位置,其中k
是非负数。进阶:
尽可能想出更多的解决方案,至少有 三种 不同的方法可以解决这个问题。
你可以使用空间复杂度为 的 原地 算法解决这个问题吗? -
示例 1:
1
2
3
4
5
6输入: nums = [1,2,3,4,5,6,7], k = 3
输出: [5,6,7,1,2,3,4]
解释:
向右轮转 1 步: [7,1,2,3,4,5,6]
向右轮转 2 步: [6,7,1,2,3,4,5]
向右轮转 3 步: [5,6,7,1,2,3,4]示例 2:
1
2
3
4
5输入:nums = [-1,-100,3,99], k = 2
输出:[3,99,-1,-100]
解释:
向右轮转 1 步: [99,-1,-100,3]
向右轮转 2 步: [3,99,-1,-100]
1 | class Solution { |
时间复杂度:.
空间复杂度:.
1 | class Solution { |
时间复杂度:.
空间复杂度:.
16. 除自身以外数组的乘积
-
题面:
给你一个整数数组
nums
,返回 数组answer
,其中answer[i]
等于nums
中除nums[i]
之外其余各元素的乘积 。题目数据 保证 数组
nums
之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。请 **不要使用除法,**且在
O(n)
时间复杂度内完成此题。进阶:你可以在 O(1) 的额外空间复杂度内完成这个题目吗?( 出于对空间复杂度分析的目的,输出数组 不被视为 额外空间。)
-
示例 1:
1
2输入: nums = [1,2,3,4]
输出: [24,12,8,6]示例 2:
1
2输入: nums = [-1,1,0,-3,3]
输出: [0,0,9,0,0]
1 | class Solution { |
时间复杂度:.
空间复杂度:.
但是违规了,题目要求不能除法。
1 | class Solution { |
时间复杂度:.
空间复杂度:.
1 | class Solution { |
时间复杂度:.
空间复杂度:.
17. 缺失的第一个正数
-
题面:
给你一个未排序的整数数组
nums
,请你找出其中没有出现的最小的正整数。请你实现时间复杂度为
O(n)
并且只使用常数级别额外空间的解决方案。 -
示例 1:
1
2
3输入:nums = [1,2,0]
输出:3
解释:范围 [1,2] 中的数字都在数组中。示例 2:
1
2
3输入:nums = [3,4,-1,1]
输出:2
解释:1 在数组中,但 2 没有。示例 3:
1
2
3输入:nums = [7,8,9,11,12]
输出:1
解释:最小的正数 1 没有出现。
1 | class Solution { |
时间复杂度:.
空间复杂度:.
1 | class Solution { |
时间复杂度:.
空间复杂度:.
1 | class Solution { |
时间复杂度:.
空间复杂度:.
1 | class Solution { |
时间复杂度:.
空间复杂度:.
18. 矩阵置零
-
题面:
给定一个 m x n 的矩阵,如果一个元素为 0 ,则将其所在行和列的所有元素都设为 0 。
请使用 原地 算法。
进阶:
- 一个直观的解决方案是使用
O(m*n)
的额外空间,但这并不是一个好的解决方案。 - 一个简单的改进方案是使用
O(m + n)
的额外空间,但这仍然不是最好的解决方案。 - 你能想出一个仅使用常量空间的解决方案吗?
- 一个直观的解决方案是使用
-
示例 1:
1
2输入:matrix = [[1,1,1],[1,0,1],[1,1,1]]
输出:[[1,0,1],[0,0,0],[1,0,1]]示例 2:
1
2输入:matrix = [[0,1,2,0],[3,4,5,2],[1,3,1,5]]
输出:[[0,0,0,0],[0,4,5,0],[0,3,1,0]]
1 | class Solution { |
时间复杂度:.
空间复杂度:.
19. 螺旋矩阵
-
题面:
给你一个 m 行 n 列的矩阵 matrix ,请按照 顺时针螺旋顺序 ,返回矩阵中的所有元素。
-
示例 1:
1
2输入:matrix = [[1,2,3],[4,5,6],[7,8,9]]
输出:[1,2,3,6,9,8,7,4,5]示例 2:
1
2输入:matrix = [[1,2,3,4],[5,6,7,8],[9,10,11,12]]
输出:[1,2,3,4,8,12,11,10,9,5,6,7]
1 | class Solution { |
时间复杂度:.
空间复杂度:.
20. 旋转图像
-
题面:
给定一个 n × n 的二维矩阵 matrix 表示一个图像。请你将图像顺时针旋转 90 度。
你必须在 原地 旋转图像,这意味着你需要直接修改输入的二维矩阵。请不要 使用另一个矩阵来旋转图像。
-
示例 1:
1
2输入:matrix = [[1,2,3],[4,5,6],[7,8,9]]
输出:[[7,4,1],[8,5,2],[9,6,3]]示例 2:
1
2输入:matrix = [[5,1,9,11],[2,4,8,10],[13,3,6,7],[15,14,12,16]]
输出:[[15,13,2,5],[14,3,4,1],[12,6,8,9],[16,7,10,11]]
1 | class Solution { |
时间复杂度:.
空间复杂度:.
1 | class Solution { |
时间复杂度:.
空间复杂度:.
21. 搜索二维矩阵 II
-
题面:
编写一个高效的算法来搜索 m x n 矩阵 matrix 中的一个目标值 target 。该矩阵具有以下特性:
- 每行的元素从左到右升序排列。
- 每列的元素从上到下升序排列。
-
示例 1:
1
2输入:matrix = [[1,4,7,11,15],[2,5,8,12,19],[3,6,9,16,22],[10,13,14,17,24],[18,21,23,26,30]], target = 5
输出:true示例 2:
1
2输入:matrix = [[1,4,7,11,15],[2,5,8,12,19],[3,6,9,16,22],[10,13,14,17,24],[18,21,23,26,30]], target = 20
输出:false
1 | class Solution { |
时间复杂度:.
空间复杂度:.
1 | class Solution { |
时间复杂度:.
空间复杂度:.
1 | class Solution { |
时间复杂度:.
空间复杂度:.
六、链表
22. 相交链表
-
题面:
给你两个单链表的头节点 headA 和 headB ,请你找出并返回两个单链表相交的起始节点。如果两个链表不存在相交节点,返回 null 。
图示两个链表在节点 c1 开始相交:
题目数据 保证 整个链式结构中不存在环。
注意,函数返回结果后,链表必须 保持其原始结构 。
1 | /** |
时间复杂度:.
空间复杂度:.
1 | /** |
时间复杂度:.
空间复杂度:.
23. 反转链表
-
题面:
给你单链表的头节点 head ,请你反转链表,并返回反转后的链表。 -
示例 1:
1 | /** |
时间复杂度:.
空间复杂度:.
24. 回文链表
-
题面:
给你一个单链表的头节点 head ,请你判断该链表是否为回文链表。如果是,返回 true ;否则,返回 false 。 -
示例 1:
1 | /** |
时间复杂度:.
空间复杂度:.
25. 环形链表
-
题面:
给你一个链表的头节点 head ,判断链表中是否有环。如果链表中存在环 ,则返回 true 。 否则,返回 false 。
1 | /** |
时间复杂度:.
空间复杂度:.
1 | /** |
时间复杂度:.
空间复杂度:.
26. 环形链表 II
-
题面:
给定一个链表的头节点 head ,返回链表开始入环的第一个节点。 如果链表无环,则返回 null。
不允许修改 链表。
1 | /** |
时间复杂度:.
空间复杂度:.
1 | /** |
时间复杂度:.
空间复杂度:.
27. 合并两个有序链表
-
题面:
将两个升序链表合并为一个新的 升序 链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。
1 | /** |
时间复杂度:.
空间复杂度:.
1 | /** |
时间复杂度:.
空间复杂度:.
28. 两数相加
-
题面:
给你两个 非空 的链表,表示两个非负的整数。它们每位数字都是按照 逆序 的方式存储的,并且每个节点只能存储 一位 数字。请你将两个数相加,并以相同形式返回一个表示和的链表。
-
示例:
1
2
3输入:l1 = [2,4,3], l2 = [5,6,4]
输出:[7,0,8]
解释:342 + 465 = 807.
1 | /** |
时间复杂度:.
空间复杂度:.
29. 删除链表的倒数第 N 个结点
-
题面:
给你一个链表,删除链表的倒数第 n 个结点,并且返回链表的头结点。
-
示例:
1
2输入:head = [1,2,3,4,5], n = 2
输出:[1,2,3,5]
1 | /** |
时间复杂度:.
空间复杂度:.
1 | /** |
时间复杂度:.
空间复杂度:.
1 | /** |
时间复杂度:.
空间复杂度:.
30. 两两交换链表中的节点
-
题面:
给你一个链表,两两交换其中相邻的节点,并返回交换后链表的头节点。你必须在不修改节点内部的值的情况下完成本题(即,只能进行节点交换)。
-
示例:
1 | /** |
时间复杂度:.
空间复杂度:.
1 | /** |
时间复杂度:.
空间复杂度:.
31. K 个一组翻转链表
-
题面:
给你链表的头节点 head ,每 k 个节点一组进行翻转,请你返回修改后的链表。
k 是一个正整数,它的值小于或等于链表的长度。如果节点总数不是 k 的整数倍,那么请将最后剩余的节点保持原有顺序。
你不能只是单纯的改变节点内部的值,而是需要实际进行节点交换。
-
示例:
1 | /** |
时间复杂度:.
空间复杂度:.
32. 随机链表的复制
-
题面:
给你一个长度为 n 的链表,每个节点包含一个额外增加的随机指针 random ,该指针可以指向链表中的任何节点或空节点。
构造这个链表的 深拷贝。 深拷贝应该正好由 n 个 全新 节点组成,其中每个新节点的值都设为其对应的原节点的值。新节点的 next 指针和 random 指针也都应指向复制链表中的新节点,并使原链表和复制链表中的这些指针能够表示相同的链表状态。复制链表中的指针都不应指向原链表中的节点 。
例如,如果原链表中有 X 和 Y 两个节点,其中 X.random --> Y 。那么在复制链表中对应的两个节点 x 和 y ,同样有 x.random --> y 。如果不指向任何节点,则为 null 。
返回复制链表的头节点。
你的代码 只 接受原链表的头节点 head 作为传入参数。
-
示例:
1 | /* |
时间复杂度:.
空间复杂度:.
33. 排序链表
-
题面:
给你链表的头结点 head ,请将其按 升序 排列并返回 排序后的链表 。
-
示例:
插入排序,超出时间限制。
1 | /** |
时间复杂度:.
空间复杂度:.
归并排序,可以。
1 | /** |
时间复杂度:.
空间复杂度:.
注:若改为自底向上归并排序,还可以空间复杂度降为 ,但是没必要。
参考:链接
耍赖也未尝不可!
1 | /** |
时间复杂度:.
空间复杂度:.
34. 合并 K 个升序链表
-
题面:
给你一个链表数组,每个链表都已经按升序排列。
请你将所有链表合并到一个升序链表中,返回合并后的链表。
-
示例:
1
2
3
4
5
6
7
8
9
10输入:lists = [[1,4,5],[1,3,4],[2,6]]
输出:[1,1,2,3,4,4,5,6]
解释:链表数组如下:
[
1->4->5,
1->3->4,
2->6
]
将它们合并到一个有序链表中得到。
1->1->2->3->4->4->5->6
1 | /** |
时间复杂度:.
空间复杂度:.
耍赖未尝不可。
1 | /** |
时间复杂度:.
空间复杂度:.
35. LRU 缓存
-
题面:
请你设计并实现一个满足 LRU (最近最少使用) 缓存 约束的数据结构。
实现LRUCache
类:
LRUCache(int capacity)
以 正整数 作为容量capacity
初始化 LRU 缓存
int get(int key)
如果关键字key
存在于缓存中,则返回关键字的值,否则返回-1
。
void put(int key, int value)
如果关键字key
已经存在,则变更其数据值value
;如果不存在,则向缓存中插入该组key-value
。如果插入操作导致关键字数量超过capacity
,则应该 逐出 最久未使用的关键字。
函数get
和put
必须以O(1)
的平均时间复杂度运行。 -
示例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17输入
["LRUCache", "put", "put", "get", "put", "get", "put", "get", "get", "get"]
[[2], [1, 1], [2, 2], [1], [3, 3], [2], [4, 4], [1], [3], [4]]
输出
[null, null, null, 1, null, -1, null, -1, 3, 4]
解释
LRUCache lRUCache = new LRUCache(2);
lRUCache.put(1, 1); // 缓存是 {1=1}
lRUCache.put(2, 2); // 缓存是 {1=1, 2=2}
lRUCache.get(1); // 返回 1
lRUCache.put(3, 3); // 该操作会使得关键字 2 作废,缓存是 {1=1, 3=3}
lRUCache.get(2); // 返回 -1 (未找到)
lRUCache.put(4, 4); // 该操作会使得关键字 1 作废,缓存是 {4=4, 3=3}
lRUCache.get(1); // 返回 -1 (未找到)
lRUCache.get(3); // 返回 3
lRUCache.get(4); // 返回 4
1 | // 本题需要设计数据结构 |
时间复杂度:put
和get
均为.
空间复杂度:.
七、二叉树
36. 二叉树的中序遍历
-
题面:
给定一个二叉树的根节点 root ,返回 它的 中序 遍历 。
-
示例:
1
2输入:root = [1,null,2,3]
输出:[1,3,2]
1 | /** |
时间复杂度:.
空间复杂度:.
37. 二叉树的最大深度
-
题面:
给定一个二叉树
root
,返回其最大深度。二叉树的 最大深度 是指从根节点到最远叶子节点的最长路径上的节点数。
-
示例:
1
2输入:root = [3,9,20,null,null,15,7]
输出:3
DFS
1 | /** |
时间复杂度:.
空间复杂度:.
BFS
1 | /** |
时间复杂度:.
空间复杂度:.
38. 翻转二叉树
-
题面:
给你一棵二叉树的根节点
root
,翻转这棵二叉树,并返回其根节点。 -
示例:
1 | /** |
时间复杂度:.
空间复杂度:.
39. 对称二叉树
-
题面:
给你一个二叉树的根节点
root
, 检查它是否轴对称。 -
示例:
自己想的使用栈的方法。
1 | /** |
时间复杂度:.
空间复杂度:.
递归方法,复杂度是一样的,但是逻辑更巧妙。
1 | /** |
时间复杂度:.
空间复杂度:.
40. 二叉树的直径
-
题面:
给你一棵二叉树的根节点,返回该树的
直径
。二叉树的 直径 是指树中任意两个节点之间最长路径的
长度
。这条路径可能经过也可能不经过根节点root
。两节点之间路径的
长度
由它们之间边数表示。 -
示例:
1
2
3输入:root = [1,2,3,4,5]
输出:3
解释:3 ,取路径 [4,2,1,3] 或 [5,2,1,3] 的长度。
1 | /** |
时间复杂度:.
空间复杂度:.
41. 二叉树的层序遍历
-
题面:
给你二叉树的根节点
root
,返回其节点值的层序遍历
。 (即逐层地,从左到右访问所有节点)。 -
示例:
1
2输入:root = [3,9,20,null,null,15,7]
输出:[[3],[9,20],[15,7]]
1 | /** |
时间复杂度:.
空间复杂度:.
1 | /** |
时间复杂度:.
空间复杂度:.
42. 将有序数组转换为二叉搜索树
-
题面:
给你一个整数数组
nums
,其中元素已经按升序
排列,请你将其转换为一棵 平衡 二叉搜索树。平衡二叉树
是指该树所有节点的左右子树的深度相差不超过 1。 -
示例:
1
2
3输入:nums = [-10,-3,0,5,9]
输出:[0,-3,9,-10,null,5]
解释:[0,-10,5,null,-3,null,9] 也将被视为正确答案:
1 | /** |
时间复杂度:.
空间复杂度:.
1 | /** |
时间复杂度:.
空间复杂度:.
43. 验证二叉搜索树
-
题面:
给你一个二叉树的根节点
root
,判断其是否是一个有效的二叉搜索树。有效
二叉搜索树
定义如下:-
节点的左子树只包含 小于 当前节点的数。
-
节点的右子树只包含 大于 当前节点的数。
-
所有左子树和右子树自身必须也是二叉搜索树。
-
-
示例1:
1
2输入:root = [2,1,3]
输出:true示例2:
1
2
3输入:root = [5,1,4,null,null,3,6]
输出:false
解释:根节点的值是 5 ,但是右子节点的值是 4 。
自己想的,感觉通俗易懂。
1 | /** |
时间复杂度:.
空间复杂度:.
递归
1 | /** |
时间复杂度:.
空间复杂度:.
用栈实现的中序遍历,不如我想的方法,复杂度一样。
但是栈实现中序遍历的思想要学会。
1 | /** |
时间复杂度:.
空间复杂度:.
44. 二叉搜索树中第 K 小的元素
-
题面:
给定一个二叉搜索树的根节点
root
,和一个整数k
,请你设计一个算法查找其中第k
小的元素(从 1 开始计数)。 -
示例 1:
1
2输入:root = [3,1,4,null,2], k = 1
输出:1示例 2:
1
2输入:root = [5,3,6,2,4,null,null,1], k = 3
输出:3
1 | /** |
时间复杂度:.
空间复杂度:.
1 | /** |
时间复杂度:.
空间复杂度:.
45. 二叉树的右视图
-
题面:
给定一个二叉树的 根节点
root
,想象自己站在它的右侧,按照从顶部到底部的顺序,返回从右侧所能看到的节点值。 -
示例:
1
2输入: [1,2,3,null,5,null,4]
输出: [1,3,4]
1 | /** |
时间复杂度:.
空间复杂度:.
46. 二叉树展开为链表
-
题面:
给你二叉树的根结点
root
,请你将它展开为一个单链表
:-
展开后的单链表应该同样使用
TreeNode
,其中right
子指针指向链表中下一个结点,而左子指针始终为null
。 -
展开后的单链表应该与二叉树
先序遍历
顺序相同。
-
-
示例:
1
2输入:root = [1,2,5,3,4,null,6]
输出:[1,null,2,null,3,null,4,null,5,null,6]
1 | class Solution { |
时间复杂度:.
空间复杂度:.
1 | class Solution { |
时间复杂度:.
空间复杂度:.
1 | class Solution { |
时间复杂度:.
空间复杂度:.
1 | class Solution { |
时间复杂度:.
空间复杂度:.
47. 从前序与中序遍历序列构造二叉树
-
题面:
给定两个整数数组
preorder
和inorder
,其中preorder
是二叉树的先序遍历,inorder
是同一棵树的中序遍历,请构造二叉树并返回其根节点。preorder
和inorder
均 无重复 元素inorder
均出现在preorder
preorder
保证 为二叉树的前序遍历序列inorder
保证 为二叉树的中序遍历序列
-
示例:
1
2输入: preorder = [3,9,20,15,7], inorder = [9,3,15,20,7]
输出: [3,9,20,null,null,15,7]
1 | /** |
时间复杂度:.
空间复杂度:.
1 | /** |
时间复杂度:.
空间复杂度:.
48. 路径总和 III
-
题面:
给定一个二叉树的根节点root
,和一个整数targetSum
,求该二叉树里节点值之和等于targetSum
的路径
的数目。路径
不需要从根节点开始,也不需要在叶子节点结束,但是路径方向必须是向下的(只能从父节点到子节点)。 -
示例:
1
2
3输入:root = [10,5,-3,3,2,null,11,3,-2,null,1], targetSum = 8
输出:3
解释:和等于 8 的路径有 3 条,如图所示。
1 | /** |
时间复杂度:.
空间复杂度:.
借用哈希表,存储前面已经计算的结果。
每次借用前缀和出现的次数,来寻找本次能满足条件的数量。
1 | /** |
时间复杂度:.
空间复杂度:.
49. 二叉树的最近公共祖先
-
题面:
给定一个二叉树, 找到该树中两个指定节点的最近公共祖先。百度百科中最近公共祖先的定义为:“对于有根树 T 的两个节点 p、q,最近公共祖先表示为一个节点 x,满足 x 是 p、q 的祖先且 x 的深度尽可能大(一个节点也可以是它自己的祖先)。”
-
示例 1:
1
2
3输入:root = [3,5,1,6,2,0,8,null,null,7,4], p = 5, q = 1
输出:3
解释:节点 5 和节点 1 的最近公共祖先是节点 3 。示例 2:
1
2
3输入:root = [3,5,1,6,2,0,8,null,null,7,4], p = 5, q = 4
输出:5
解释:节点 5 和节点 4 的最近公共祖先是节点 5 。因为根据定义最近公共祖先节点可以为节点本身。
1 | /** |
时间复杂度:.
空间复杂度:.
哈希表存储父节点。
1 | /** |
时间复杂度:.
空间复杂度:.
50. 二叉树中的最大路径和
-
题面:
二叉树中的
路径
被定义为一条节点序列,序列中每对相邻节点之间都存在一条边。同一个节点在一条路径序列中 至多出现一次 。该路径 至少包含一个 节点,且不一定经过根节点。路径和
是路径中各节点值的总和。给你一个二叉树的根节点
root
,返回其最大路径和
。 -
示例:
1
2
3输入:root = [-10,9,20,null,null,15,7]
输出:42
解释:最优路径是 15 -> 20 -> 7 ,路径和为 15 + 20 + 7 = 42
1 | /** |
时间复杂度:.
空间复杂度:.
51.
- 题面:
1 |
时间复杂度:.
空间复杂度:.
1 |
时间复杂度:.
空间复杂度:.