脑子越来越不好使,刷点算法题提高点智商。
Medium
每日温度
题目内容
解题思路
遍历每日温度,维护一个单调栈:
- 如果栈为空或当日温度小于等于栈顶元素,则直接入栈。
- 如果栈不为空并且当日温度大于栈顶元素,说明栈顶元素的升温日找到了,出栈并计算天数,继续判断栈顶元素。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
| const dailyTemperatures = (temperatures: number[]): number[] => {
let result: number[] = new Array(temperatures.length).fill(0);
let stack: number[] = [0];
for (let i = 1; i < temperatures.length; i++) {
while (
stack.length &&
temperatures[i] > temperatures[stack[stack.length - 1]]
) {
const top: number = stack.pop();
result[top] = i - top;
}
stack.push(i);
}
return result;
};
|
任务调度器
题目内容
解题思路
贪心。
先排个数最多的任务 A,在 A 的冷却时间内插入其他任务,先计算前 n-1 行 n 的间隔的时间大小,再计算和最大次数相同的字母个数,然后累加进 ret。最后在 tasks 的长度和 ret 中取较大的一个。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
| const leastInterval = (tasks: string[], n: number): number => {
let arr: number[] = new Array(26).fill(0);
for (let c of tasks) {
arr[c.charCodeAt(0) - "A".charCodeAt(0)]++;
}
let max: number = Math.max(...arr);
let ret: number = (max - 1) * (n + 1);
for (let i = 0; i < 26; i++) {
if (arr[i] === max) {
ret++;
}
}
return Math.max(ret, tasks.length);
};
|
LRU 缓存
题目内容
解题思路
Map 迭代器。
- Map 中的键值对按照插入的顺序排列,第一个插入的键值对会排在第一个位置,第二个插入的键值对会排在第二个位置,以此类推。
- map.keys().next().value 会返回 Map 对象中的第一个键值对的键。
1
2
3
4
5
6
| const myMap: Map<string, string> = new Map();
myMap.set("key1", "value1");
myMap.set("key2", "value2");
myMap.set("key3", "value3");
console.log(myMap.keys().next().value); // 输出: key1
|
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
| class LRUCache {
capacity: number;
map: Map<number, number>;
constructor(capacity: number) {
this.capacity = capacity;
this.map = new Map();
}
get(key: number): number {
if (this.map.has(key)) {
let value = this.map.get(key);
this.map.delete(key);
this.map.set(key, value);
return value;
}
return -1;
}
put(key: number, value: number): void {
if (this.map.has(key)) {
this.map.delete(key);
}
this.map.set(key, value);
if (this.map.size > this.capacity) {
this.map.delete(this.map.keys().next().value);
}
}
}
/**
* Your LRUCache object will be instantiated and called as such:
* var obj = new LRUCache(capacity)
* var param_1 = obj.get(key)
* obj.put(key,value)
*/
|
根据身高重建队列
题目内容
解题思路
贪心。
按照身高 h 来排序,身高一定是从大到小排(身高相同的话则 k 小的站前面),让高个子在前面。
然后按照 k 为下标重新插入队列,插入操作过后的 queue 满足队列属性。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
| const reconstructQueue = (people: number[][]): number[][] => {
let queue: number[][] = [];
people.sort((a, b) => {
if (b[0] !== a[0]) {
return b[0] - a[0];
} else {
return a[1] - b[1];
}
});
for (let i = 0; i < people.length; i++) {
queue.splice(people[i][1], 0, people[i]);
}
return queue;
};
|
打家劫舍
题目内容
解题思路
动态规划。
五部曲:
- 确定 dp 数组及下标含义:dp[i]:下标 i(包括 i) 以内的房屋,最多可以偷窃的金额为 dp[i]。
- 确定递推公式:dp[i] = Math.max(dp[i - 2] + nums[i], dp[i - 1])。
决定 dp[i] 的因素是第 i 个房间偷还是不偷:- 偷:dp[i] = dp[i - 2] + nums[i]。
- 不偷:dp[i] = dp[i -1],考虑 i - 1 房。
- dp 数组初始化:dp[0] = nums[0],dp[1] = Math.max(nums[0], nums[1])。
- 确定遍历顺序:从前到后遍历,i 初始化为 2。
- 举例推导验证。
代码实现
1
2
3
4
5
6
7
8
| const rob = (nums: number[]): number => {
const len: number = nums.length;
const dp = [nums[0], Math.max(nums[0], nums[1])];
for (let i = 2; i < len; ++i) {
dp[i] = Math.max(nums[i] + dp[i - 2], dp[i - 1]);
}
return dp[len - 1];
};
|
子集
题目内容
解题思路
枚举出当前可选的数:
- 如果第一个数选 1,选第二个数,2、3 可选;
- 如果第一个数选 2,选第二个数,只有 3 可选;
- 如果第一个数选 3,没有第二个数可选;
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
| /**
Do not return anything, modify nums in-place instead.
*/
const subsets = (nums: number[]): number[][] => {
const result: number[][] = [];
const dfs = (index: number, list: number[]) => {
result.push(list.slice());
for (let i = index; i < nums.length; i++) {
list.push(nums[i]);
dfs(i + 1, list);
list.pop();
}
};
dfs(0, []);
return result;
};
|
颜色分类
题目内容
解题思路
荷兰国旗问题,三指针算法。
使用三个指针(low、mid 和 high)来遍历和排序数组。执行过程中,所有位于 low 指针左侧的元素都是红色,位于 high 指针右侧的元素都是蓝色,确保所有白色元素位于 low 和 high 指针之间。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
| /**
Do not return anything, modify nums in-place instead.
*/
const sortColors = (nums: number[]): void => {
let low: number = 0,
mid: number = 0,
high: number = nums.length - 1;
while (mid <= high) {
switch (nums[mid]) {
case 0:
[nums[mid], nums[low]] = [nums[low], nums[mid]];
low++;
mid++;
break;
case 1:
mid++;
break;
case 2:
[nums[mid], nums[high]] = [nums[high], nums[mid]];
high--;
break;
default:
return;
}
}
};
|
最小路径和
题目内容
解题思路
动态规划。
从左上角起始点到右下角终点不断累加找到最短路径。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
| const minPathSum = (grid: number[][]): number => {
const m: number = grid.length,
n: number = grid[0].length;
const dp: number[][] = new Array(m).fill(0).map((_) => new Array(n).fill(0));
dp[0][0] = grid[0][0];
for (let i = 0; i < m; i++) {
for (let j = 0; j < n; j++) {
switch (true) {
case i === 0 && j !== 0:
dp[i][j] = grid[i][j] + dp[i][j - 1];
break;
case i !== 0 && j === 0:
dp[i][j] = grid[i][j] + dp[i - 1][j];
break;
case i !== 0 && j !== 0:
dp[i][j] = grid[i][j] + Math.min(dp[i - 1][j], dp[i][j - 1]);
break;
default:
return;
}
}
}
return dp[m - 1][n - 1];
};
|
不同路径
题目内容
解题思路
动态规划。
每个位置的路径 = 该位置左边的路径 + 该位置上边的路径。
代码实现
1
2
3
4
5
6
7
8
9
| const uniquePaths = (m: number, n: number): number => {
const dp = new Array(m).fill(1).map((i) => new Array(n).fill(1));
for (let i = 1; i < m; ++i) {
for (let j = 1; j < n; ++j) {
dp[i][j] = dp[i - 1][j] + dp[i][j - 1];
}
}
return dp[m - 1][n - 1];
};
|
合并区间
题目内容
解题思路
prev 初始化为第一个区间,curr 为当前遍历到的区间, res 为结果数组。尝试合并 prev 和 curr,合并后更新到 prev,合并后的新区间还可能会和后面的区间重合,继续尝试合并新的 curr,更新给 prev,直到不能合并,将 prev 推入到 res。先合并遇到不重合再推入 prev,当考察完最后一个区间,遇不到不重合的区间,要单独把 prev 推入到 res。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
| const merge = (intervals: number[][]): number[][] => {
let res: number[][] = [];
intervals.sort((a: number[], b: number[]) => a[0] - b[0]);
let prev: number[] = intervals[0];
for (let i = 1; i < intervals.length; ++i) {
let curr = intervals[i];
if (prev[1] >= curr[0]) {
prev[1] = Math.max(prev[1], curr[1]);
} else {
res.push(prev);
prev = curr;
}
}
res.push(prev);
return res;
};
|
跳跃游戏
题目内容
解题思路
贪心。
重点看可跳跃的覆盖范围,在可跳跃范围内,每次取最大跳跃步数更新可跳跃范围,如果更新后的可跳跃覆盖范围大于等于终点下标,则直接返回 true,否则只前进一步继续更新可跳跃范围。
代码实现
1
2
3
4
5
6
7
8
9
| const canJump = (nums: number[]): boolean => {
if (nums.length === 1) return true;
let cover: number = 0;
for (let i = 0; i <= cover; ++i) {
cover = Math.max(nums[i] + i, cover);
if (cover >= nums.length - 1) return true;
}
return false;
};
|
最大子数组和
题目内容
解题思路
动态规划。
- 对数组进行遍历,当前最大子序列和为 sum,结果为 ans。
- sum > 0,则说明 sum 对结果有增益,保留 sum 并加上当前遍历数字。
- sum < 0,则说明 sum 对结果无增益,需要舍弃,sum 更新为当前遍历数字。
- 每次遍历比较 sum 和 ans 的大小,将最大值置为 ans。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
| const maxSubArray = (nums: number[]): number => {
let ans: number = nums[0];
let sum: number = 0;
for (let num of nums) {
if (sum > 0) {
sum += num;
} else {
sum = num;
}
ans = Math.max(ans, sum);
}
return ans;
};
|
字母异位词分组
题目内容
解题思路
哈希表。
代码实现
1
2
3
4
5
6
7
8
9
10
11
| const groupAnagrams = (strs: string[]): string[][] => {
let map: Map<string, string[]> = new Map();
for (let i = 0; i < strs.length; i++) {
let key: string = strs[i].split("").sort().join("");
if (!map.has(key)) {
map.set(key, []);
}
map.get(key).push(strs[i]);
}
return [...map.values()];
};
|
旋转图像
题目内容
解题思路
两次翻转。
matrix = [[5,1,9,11],[2,4,8,10],[13,3,6,7],[15,14,12,16]]
先根据水平轴翻转得到:
[[15,14,12,16],[13,3,6,7],[2,4,8,10],[5,1,9,11]]
再根据主对角线翻转得到:
[[15,13,2,5],[14,3,4,1],[12,6,8,9],[16,7,10,11]]
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
| /**
Do not return anything, modify matrix in-place instead.
*/
const rotate = (matrix: number[][]): void => {
const n: number = matrix.length;
for (let i = 0; i < Math.floor(n / 2); ++i) {
for (let j = 0; j < n; ++j) {
[matrix[i][j], matrix[n - i - 1][j]] = [
matrix[n - i - 1][j],
matrix[i][j],
];
}
}
for (let i = 0; i < n; ++i) {
for (let j = 0; j < i; ++j) {
[matrix[i][j], matrix[j][i]] = [matrix[j][i], matrix[i][j]];
}
}
};
|
全排列
题目内容
解题思路
回溯 + 剪枝。
- 每一位都有 3 种选择:1、2、3。
- 每一次都做选择,展开出一棵空间树。
- 利用 hashMap,记录选过的数,下次遇到相同的数,跳过。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
| const permute = (nums: number[]): number[][] => {
const res: number[][] = [];
const used: Record<string, boolean> = {};
const dfs = (path: number[]) => {
if (path.length === nums.length) {
res.push(path.slice());
return;
}
for (const num of nums) {
if (used[String(num)]) continue;
path.push(num);
used[String(num)] = true;
dfs(path);
path.pop();
used[String(num)] = false;
}
};
dfs([]);
return res;
};
|
组合总和
题目内容
解题思路
回溯 + 剪枝。
- 剪枝:sum > target 结束当前递龟;sum === target 路径加入解集;限制下次递龟的起点避免重复组合。
- ×:当前组合和之前生成的组合重复了。
- △:当前求和 > target,不能选下去了,返回。
- ○:求和正好 == target,加入解集,并返回。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
| const combinationSum = (candidates: number[], target: number): number[][] => {
const res: number[][] = [];
const backTrace = (start: number, temp: number[], sum: number): void => {
if (sum > target) return;
if (sum === target) res.push([...temp]);
for (let i = start; i < candidates.length; i++) {
temp.push(candidates[i]); // 选这个数
backTrace(i, temp, sum + candidates[i]);
temp.pop(); // 撤销选择,继续尝试选同层右边的数
}
};
backTrace(0, [], 0);
return res;
};
|
在排序数组中查找元素的第一个和最后一个位置
题目内容
解题思路
题目要求时间复杂度为 O(log n),二分法。
- 二分查找中,寻找 leftIndex 即为在数组中寻找第一个大于等于 target 的下标,寻找 rightIndex 即为在数组中寻找第一个大于 target 的下标,然后将下标减一。
- binarySearch(nums, target, lower) 表示在 nums 数组中二分查找 target 的位置,如果 lower 为 true,则查找第一个大于等于 target 的下标,否则查找第一个大于 target 的下标。
- target 可能不存在数组中,需要重新校验获得的两个下标 leftIndex 和 rightIndex,不符合则返回 [-1, -1];
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
| const searchRange = (nums: number[], target: number): number[] => {
const binarySearch = (
nums: number[],
target: number,
lower: boolean
): number => {
let left: number = 0,
right: number = nums.length - 1,
ans: number = nums.length;
while (left <= right) {
const mid: number = left + ((right - left) >> 1);
if (nums[mid] > target || (lower && nums[mid] >= target)) {
right = mid - 1;
ans = mid;
} else {
left = mid + 1;
}
}
return ans;
};
let ans = [-1, -1];
const leftIndex = binarySearch(nums, target, true);
const rightIndex = binarySearch(nums, target, false) - 1;
if (
leftIndex <= rightIndex &&
rightIndex < nums.length &&
nums[leftIndex] === target &&
nums[rightIndex] === target
) {
ans = [leftIndex, rightIndex];
}
return ans;
};
|
搜索旋转排序数组
题目内容
解题思路
题目要求时间复杂度为 O(log n),二分法。
随便选择一个点,将数组一分为二,其中一部分一定是有序的。
- 先找出 mid,根据 mid 来判断,mid 是在有序部分还是无序部分。
- 如果 mid 小于 start,则 mid 一定在右边有序部分。
- 如果 mid 大于等于 start, 则 mid 一定在左边有序部分。
- 继续判断 target 在哪一部分。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
| const search = (nums: number[], target: number): number => {
let start: number = 0;
let end: number = nums.length - 1;
while (start <= end) {
const mid: number = start + ((end - start) >> 1);
if (nums[mid] === target) return mid;
if (nums[mid] >= nums[start]) {
if (target >= nums[start] && target <= nums[mid]) {
end = mid - 1;
} else {
start = mid + 1;
}
} else {
if (target >= nums[mid] && target <= nums[end]) {
start = mid + 1;
} else {
end = mid - 1;
}
}
}
return -1;
};
|
下一个排列
题目内容
解题思路
从低位挑一个大一点的数,交换前面一个小一点的数,变大的幅度要尽量小。
- [3, 2, 1],没有下一个排列,已经稳定了,没法变大了。
- [1, 5, 2, 4, 3, 2],从右往左,寻找第一个比右邻居小的数,把它换到后面去。
- 1 5 (2) 4 3 2,找到中间这个 (2),接着还是从右往左找,找第一个比 2 大的数和它交换,交换后:1 5 (3) 4 (2) 2。
- 此时还没结束,变大的幅度还可以再小一点,千位变大了,后三位可以继续再小一点。
- 后三位肯定是递减的,翻转,变为 [1, 5, 3, 2, 2, 4],即为所求。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
| /**
Do not return anything, modify nums in-place instead.
*/
const nextPermutation = (nums: number[]): void => {
let i: number = nums.length - 2;
while (i >= 0 && nums[i] >= nums[i + 1]) {
i--;
}
if (i >= 0) {
let j: number = nums.length - 1;
while (j >= 0 && nums[j] <= nums[i]) {
j--;
}
[nums[i], nums[j]] = [nums[j], nums[i]];
}
let l: number = i + 1;
let r: number = nums.length - 1;
while (l < r) {
[nums[l], nums[r]] = [nums[r], nums[l]];
l++;
r--;
}
};
|
括号生成
题目内容
解题思路
BFS + 去重。
从 n - 1 推导 n 的组合情况,只需要遍历 n - 1 的所有组合,并在所有组合的每个位置填入一对括号 () 并去重即可。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
| const generateParenthesis = (n: number): string[] => {
let set: Set<string> = new Set(["()"]);
for (let i = 2; i <= n; i++) {
let nextSet: Set<string> = new Set();
for (const s of set) {
for (let j = 0; j < s.length; j++) {
nextSet.add(`${s.slice(0, j)}()${s.slice(j)}`);
}
}
set = nextSet;
}
return [...set];
};
|
删除链表的倒数第 N 个结点
题目内容
解题思路
快慢指针 + 虚拟头结点。
- 初始化快慢指针 slow、fast 指向虚拟头结点。
- fast 向后移动 n 个位置,此时 fast 和 slow 之间相隔的元素个数为 n。
- 同时向后移动 fast 和 slow,当 fast 移动到最后时,slow 的下一个结点就是要删除的倒数第 n 个结点。
- 将 slow 的下一个结点指向下下个结点,则删除倒数第 n 个结点。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
| /**
* Definition for singly-linked list.
* class ListNode {
* val: number
* next: ListNode | null
* constructor(val?: number, next?: ListNode | null) {
* this.val = (val===undefined ? 0 : val)
* this.next = (next===undefined ? null : next)
* }
* }
*/
const removeNthFromEnd = (
head: ListNode | null,
n: number
): ListNode | null => {
let dmy: ListNode | null = new ListNode(0, head);
let slow: ListNode | null = dmy;
let fast: ListNode | null = dmy;
while (n--) fast = fast.next;
while (fast && fast.next) {
fast = fast.next;
slow = slow.next;
}
slow.next = slow.next.next;
return dmy.next;
};
|
电话号码的字母组合
题目内容
解题思路
回溯。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
| const letterCombinations = (digits: string): string[] => {
const map = [
"",
"",
"abc",
"def",
"ghi",
"jkl",
"mno",
"pqrs",
"tuv",
"wxyz",
];
const ans: string[] = [];
const length = digits.length;
if (!length) return ans;
const backTrack = (digitsIndex: number, s: string) => {
if (digitsIndex === length) {
return ans.push(s);
}
for (let i = 0; i < map[digits[digitsIndex]].length; i++) {
backTrack(digitsIndex + 1, s + map[digits[digitsIndex]][i]);
}
};
backTrack(0, "");
return ans;
};
|
三数之和
题目内容
解题思路
双指针。
先将数组 nums 由小到大排序。
如果数组长度小于 3,则直接返回空数组。
固定 3 个指针中最左侧的指针 i,双指针 left、right 分设在数组索引 i + 1 和数组尾端,交替向中间移动,记录每个固定指针 i 的所有满足 nums[i] + nums[right] + nums[left] = 0 的 i、left、right 组合:
- 当 num[i] > 0 时直接跳出,因为 nums[i] + nums[right] + nums[left] > 0,即 3 个数字都大于 0。
- 当 i > 0 且 nums[i] === nums[i - 1] 时跳过此时的元素 nums[i],因为已经将 nums[i - 1] 的所有组合加入到结果中。
- left、right 分设在数组索引两端,当 left < right 时,循环计算 nums[i] + nums[right] + nums[left]。
- 如果三者和 > 0,right–。
- 如果三者和 < 0,left++。
- 如果三者和 === 0,记录组合到 result, right–、left++、并跳过所有重复的 nums[right]、nums[left]。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
| const threeSum = (nums: number[]): number[][] => {
nums.sort((a, b) => a - b);
let result: number[][] = [];
if (nums.length < 3) return result;
for (let i = 0; i < nums.length; i++) {
if (nums[i] > 0) return result;
if (i > 0 && nums[i] === nums[i - 1]) continue;
let left: number = i + 1;
let right: number = nums.length - 1;
while (right > left) {
if (nums[i] + nums[right] + nums[left] > 0) {
right--;
} else if (nums[i] + nums[right] + nums[left] < 0) {
left++;
} else {
result.push([nums[i], nums[left], nums[right]]);
while (right > left && nums[left] === nums[left + 1]) left++;
while (right > left && nums[right] === nums[right - 1]) right--;
left++;
right--;
}
}
}
return result;
};
|
盛最多水的容器
题目内容
解题思路
双指针。
不断计算面积,更新最大面积。
代码实现
1
2
3
4
5
6
7
8
9
| const maxArea = (arr: number[]): number => {
let max: number = 0;
for (let i = 0, j = arr.length - 1; i < j; ) {
const minHeight: number = arr[i] < arr[j] ? arr[i++] : arr[j--];
const area: number = (j - i + 1) * minHeight;
max = Math.max(max, area);
}
return max;
};
|
最长回文子串
题目内容
解题思路
贪心 + 双指针。
中心扩展,遍历数组,找到连续一样的子字符串,然后逐渐向两侧扩展,直到越界或两侧不一致。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
| const longestPalindrome = (s: string): string => {
let max: number = 0; // 当前最大回文串的长度
let start: number = -1; // 当前最大回文串的起始索引
for (let i = 0; i < s.length; i++) {
let now: number = 1; // 当前回文串的长度
let l: number = i - 1; // 左侧开始遍历的指针
while (s[i] === s[i + 1]) {
// 向后找,如果后面字符与当前字符都一样,则长度 + 1
now++;
i++;
}
let r: number = i + 1; // 右侧开始遍历的指针
while (s[l] && s[r] && s[l] === s[r]) {
// 从连续字符两端开始逐渐向两侧扩展,直到遇到越界或者两者不一致
now += 2;
l--;
r++;
}
if (now > max) {
// 更新当前最大回文串的起始索引和长度
max = now;
start = l + 1;
}
}
return s?.slice(start, start + max); // 通过长度和起始索引,获得需要的字符串
};
|
无重复字符的最长子串
题目内容
解题思路
字符串转换为数组应用累加器求值。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
| const lengthOfLongestSubstring = (s: string): number => {
let max: number = 0;
if (s.length <= 1) return s.length;
s.split("").reduce<string>((acc: string, value: string) => {
const len = acc.indexOf(value);
if (len === -1) {
acc += value;
max = acc.length > max ? acc.length : max;
return acc;
} else {
acc += value;
return acc.slice(len + 1);
}
}, "");
return max;
};
|
两数相加
题目内容
解题思路
递 🐢。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
| /**
* Definition for singly-linked list.
* class ListNode {
* val: number
* next: ListNode | null
* constructor(val?: number, next?: ListNode | null) {
* this.val = (val===undefined ? 0 : val)
* this.next = (next===undefined ? null : next)
* }
* }
*/
const dfs = (
l1: ListNode | null,
l2: ListNode | null,
carry: number = 0
): ListNode => {
if (!l1 && !l2 && carry === 0) {
return null;
}
const value1 = l1?.val ?? 0;
const value2 = l2?.val ?? 0;
const sum = value1 + value2 + carry;
const node = new ListNode(sum % 10);
node.next = dfs(l1?.next ?? null, l2?.next ?? null, Math.floor(sum / 10));
return node;
};
const addTwoNumbers = (
l1: ListNode | null,
l2: ListNode | null
): ListNode | null => {
return dfs(l1, l2, 0);
};
|
Easy
合并二叉树
题目内容
解题思路
递 🐢。
同步遍历两棵树上的节点,直接在 t1 上修改。
- t1 为 null、t2 不为 null,t1 换成 t2。
- t2 为 null、t1 不为 null,t1 仍为 t1。
- t1 和 t2 都为 null,t1 仍为 t1。
- t1、t2 都存在,将 t2 的值加到 t1 上。
- 子树合并使用递龟。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
| /**
* Definition for a binary tree node.
* class TreeNode {
* val: number
* left: TreeNode | null
* right: TreeNode | null
* constructor(val?: number, left?: TreeNode | null, right?: TreeNode | null) {
* this.val = (val===undefined ? 0 : val)
* this.left = (left===undefined ? null : left)
* this.right = (right===undefined ? null : right)
* }
* }
*/
const mergeTrees = (
root1: TreeNode | null,
root2: TreeNode | null
): TreeNode | null => {
if (root1 === null && root2) {
return root2;
}
if ((root1 && root2 === null) || (root1 === null && root2 === null)) {
return root1;
}
root1.val += root2.val;
root1.left = mergeTrees(root1.left, root2.left);
root1.right = mergeTrees(root1.right, root2.right);
return root1;
};
|
二叉树的直径
题目内容
解题思路
递 🐢。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
| /**
* Definition for a binary tree node.
* class TreeNode {
* val: number
* left: TreeNode | null
* right: TreeNode | null
* constructor(val?: number, left?: TreeNode | null, right?: TreeNode | null) {
* this.val = (val===undefined ? 0 : val)
* this.left = (left===undefined ? null : left)
* this.right = (right===undefined ? null : right)
* }
* }
*/
const diameterOfBinaryTree = (root: TreeNode | null): number => {
let ans: number = 0;
const depth = (rootNode: TreeNode | null): number => {
if (!rootNode) return 0;
const L: number = depth(rootNode.left);
const R: number = depth(rootNode.right);
ans = Math.max(ans, L + R + 1); // 将每个节点最大直径(左子树深度 + 右子树深度)与当前最大值比较并取大者
return Math.max(L, R); // 返回节点深度
};
depth(root);
return ans;
};
|
汉明距离
题目内容
解题思路
异或位运算 + Brian Kernighan 算法。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
| const hammingDistance = (x: number, y: number): number => {
let s: number = x ^ y;
const countOneNums = (n: number): number => {
let count: number = 0;
while (n) {
n &= n - 1;
++count;
}
return count;
};
return countOneNums(s);
};
|
找到所有数组中消失的数字
题目内容
解题思路
哈希的数组替代法。
nums 数组中一定存在重复的数字,nums 中的数字一定属于 [1, n]。
在第一个 for 循环中更改了 nums 的部分值,对于 nums[i] 的值改变次数是由出现了多少次 i + 1 值。数 i 若存在则下标是 i - 1 这个位置的数一定大于 n,被改变了。
[4, 3, 2, 7, 8, 2, 3, 1] 在第一个 for 循环结束后是 [12, 19, 18, 15, 8, 2, 11, 9] n = 8。
nums[1] = 3 被改变了 2 次得到 19 即出现了两次 2,nums[2] = 2 被改变了 2 次变成了 18 即出现了两次 3。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
| const findDisappearedNumbers = (nums: number[]): number[] => {
const n: number = nums.length;
for (const num of nums) {
const x: number = (num - 1) % n;
nums[x] += n;
}
const ret: number[] = [];
for (const [i, num] of nums.entries()) {
if (num <= n) {
// 下标是 i,题目中要求的值为 i + 1
ret.push(i + 1);
}
}
return ret;
};
|
比特位计数
题目内容
解题思路
Brian Kernighan 算法。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
| const countBits = (n: number): number[] => {
const result: number[] = new Array(n + 1).fill(0);
const countOneNums = (n: number): number => {
let count: number = 0;
while (n) {
n &= n - 1;
++count;
}
return count;
};
for (let i = 0; i <= n; i++) {
result[i] = countOneNums(i);
}
return result;
};
|
移动零
题目内容
解题思路
双指针 + 遍历。
参考快速排序的思想,确定一个待分割的元素做中间节点 x,然后把所有小于等于 x 的元素放到 x 的左边,大于 x 的元素放到它的右边。
- 初始化两个指针 i 和 j,j 指针用于找中间点 0,i 指针用于找中间点右侧的非 0 元素。
- 如果两个指针指向元素均不为 0,整体向右移动一位。
- 如果 j 指针找到了中间点 0,i 没有找到中间点右侧的非 0 元素,则继续下一轮循环。
- 如果 j 指针找到了中间点 0,i 指针也找到了中间点右侧的非 0 元素,则通过一个临时变量来交换两个元素的值。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
| /**
Do not return anything, modify nums in-place instead.
*/
const moveZeroes = (nums: number[]): number[] => {
let j: number = 0;
for (let i = 0; i < nums.length; i++) {
switch (true) {
case nums[i] !== 0 && nums[j] !== 0:
j++;
break;
case nums[j] === 0 && nums[i] === 0:
continue;
break;
case nums[j] === 0 && nums[i] !== 0:
const tmp: number = nums[i];
nums[i] = nums[j];
nums[j++] = tmp;
break;
default:
return;
}
}
return nums;
};
|
回文链表
题目内容
解题思路
快慢指针 + 反转链表。
- 初始化快慢指针 slow、fast 指向 head,fast 每次走两步,slow 每次走一步,当 fast 走到最后,slow 位于中间位置,slow 在移动期间不断反转前半部分链表,反转后为 prev。
- 判断链表长度的奇偶性,如果是奇数 (即此时 fast 不为 null),slow 再向后移动一步,过滤掉中间的公共节点。
- prev 和 slow 此时分别为原链表的前半部分(反转后)和后半部分的头节点,循环判断 prev 和 slow 的值,都相同则返回 true,否则结束循环返回 false。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
| /**
* Definition for singly-linked list.
* class ListNode {
* val: number
* next: ListNode | null
* constructor(val?: number, next?: ListNode | null) {
* this.val = (val===undefined ? 0 : val)
* this.next = (next===undefined ? null : next)
* }
* }
*/
const isPalindrome = (head: ListNode | null): boolean => {
let fast: ListNode | null = head;
let slow: ListNode | null = head;
let prev: ListNode | null = null;
while (fast !== null && fast?.next !== null) {
fast = fast?.next?.next;
const next: ListNode | null = slow?.next;
slow.next = prev;
prev = slow;
slow = next;
}
if (fast !== null) slow = slow?.next;
while (slow !== null && prev !== null) {
if (slow?.val !== prev?.val) return false;
prev = prev?.next;
slow = slow?.next;
}
return true;
};
|
翻转二叉树
题目内容
解题思路
递 🐢。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
| /**
* Definition for a binary tree node.
* class TreeNode {
* val: number
* left: TreeNode | null
* right: TreeNode | null
* constructor(val?: number, left?: TreeNode | null, right?: TreeNode | null) {
* this.val = (val===undefined ? 0 : val)
* this.left = (left===undefined ? null : left)
* this.right = (right===undefined ? null : right)
* }
* }
*/
const invertTree = (root: TreeNode | null): TreeNode | null => {
if (root) {
[root.left, root.right] = [invertTree(root.right), invertTree(root.left)];
}
return root;
};
|
反转链表
题目内容
解题思路
迭代。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
| /**
* Definition for singly-linked list.
* class ListNode {
* val: number
* next: ListNode | null
* constructor(val?: number, next?: ListNode | null) {
* this.val = (val===undefined ? 0 : val)
* this.next = (next===undefined ? null : next)
* }
* }
*/
const reverseList = (head: ListNode | null): ListNode | null => {
let prev: ListNode | null = null;
let curr: ListNode | null = head;
while (curr) {
const next: ListNode | null = curr.next;
curr.next = prev;
prev = curr;
curr = next;
}
return prev;
};
|
多数元素
题目内容
解题思路
摩尔投票法: 两两对抗,票数抵消。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
| const majorityElement = (nums: number[]): number => {
let count: number = 1;
let major: number = nums[0];
for (let i = 1; i < nums.length; i++) {
if (count === 0) {
major = nums[i];
count = 1;
} else if (major === nums[i]) {
count += 1;
} else {
count -= 1;
}
}
return major;
};
|
相交链表
题目内容
解题思路
哈希。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
| /**
* Definition for singly-linked list.
* class ListNode {
* val: number
* next: ListNode | null
* constructor(val?: number, next?: ListNode | null) {
* this.val = (val===undefined ? 0 : val)
* this.next = (next===undefined ? null : next)
* }
* }
*/
const getIntersectionNode = (
headA: ListNode | null,
headB: ListNode | null
): ListNode | null => {
let setA = new Set<ListNode>();
let curr = headA;
while (curr !== null) {
setA.add(curr);
curr = curr.next;
}
curr = headB;
while (curr !== null) {
if (setA.has(curr)) return curr;
curr = curr.next;
}
return null;
};
|
环形链表
题目内容
解题思路
快慢指针。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
| /**
* Definition for singly-linked list.
* class ListNode {
* val: number
* next: ListNode | null
* constructor(val?: number, next?: ListNode | null) {
* this.val = (val===undefined ? 0 : val)
* this.next = (next===undefined ? null : next)
* }
* }
*/
const hasCycle = (head: ListNode | null): boolean => {
let slow: ListNode | null = head;
let fast: ListNode | null = head;
while (fast && fast.next) {
slow = slow.next;
fast = fast.next.next;
if (slow === fast) {
return true;
}
}
return false;
};
|
只出现一次的数字
题目内容
解题思路
异或位运算。
代码实现
1
2
3
4
5
6
7
8
9
| const singleNumber = (nums: number[]): number => {
let num: number = nums[0];
if (nums.length > 1) {
for (let i = 1; i < nums.length; i++) {
num ^= nums[i];
}
}
return num;
};
|
买卖股票的最佳时机
题目内容
解题思路
贪心,取最左最小值,取最右最大值,得到的差值就是最大利润。
代码实现
1
2
3
4
5
6
7
8
9
10
| const maxProfit = (prices: number[]): number => {
if (prices.length === 0) return 0;
let min: number = prices[0];
let max: number = 0;
for (let item of prices) {
min = Math.min(min, item);
max = Math.max(max, item - min);
}
return max;
};
|
二叉树的最大深度
题目内容
解题思路
dfs + 递 🐢。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
| /**
* Definition for a binary tree node.
* class TreeNode {
* val: number
* left: TreeNode | null
* right: TreeNode | null
* constructor(val?: number, left?: TreeNode | null, right?: TreeNode | null) {
* this.val = (val===undefined ? 0 : val)
* this.left = (left===undefined ? null : left)
* this.right = (right===undefined ? null : right)
* }
* }
*/
const maxDepth = (root: TreeNode | null): number => {
if (root === null) return 0;
let leftDepth = maxDepth(root?.left);
let rightDepth = maxDepth(root?.right);
return Math.max(leftDepth, rightDepth) + 1;
};
|
对称二叉树
题目内容
解题思路
递 🐢。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
| /**
* Definition for a binary tree node.
* class TreeNode {
* val: number
* left: TreeNode | null
* right: TreeNode | null
* constructor(val?: number, left?: TreeNode | null, right?: TreeNode | null) {
* this.val = (val===undefined ? 0 : val)
* this.left = (left===undefined ? null : left)
* this.right = (right===undefined ? null : right)
* }
* }
*/
const isSymmetric = (root: TreeNode | null): boolean => {
const check = (p: TreeNode | null, q: TreeNode | null): boolean => {
if (!p && !q) return true;
if (!p || !q) return false;
return p.val === q.val && check(p.left, q.right) && check(p.right, q.left);
};
return check(root, root);
};
|
二叉树的中序遍历
题目内容
解题思路
递 🐢。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
| /**
* Definition for a binary tree node.
* class TreeNode {
* val: number
* left: TreeNode | null
* right: TreeNode | null
* constructor(val?: number, left?: TreeNode | null, right?: TreeNode | null) {
* this.val = (val===undefined ? 0 : val)
* this.left = (left===undefined ? null : left)
* this.right = (right===undefined ? null : right)
* }
* }
*/
const inorderTraversal = (root: TreeNode | null): number[] => {
let res: number[] = [];
const preOrder = (root: TreeNode | null): void => {
if (!root) return;
preOrder(root?.left);
res.push(root?.val);
preOrder(root?.right);
};
preOrder(root);
return res;
};
|
合并两个有序链表
题目内容
解题思路
创建新节点。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
| /**
* Definition for singly-linked list.
* class ListNode {
* val: number
* next: ListNode | null
* constructor(val?: number, next?: ListNode | null) {
* this.val = (val===undefined ? 0 : val)
* this.next = (next===undefined ? null : next)
* }
* }
*/
const mergeTwoLists = (
list1: ListNode | null,
list2: ListNode | null
): ListNode | null => {
const head: ListNode = new ListNode(-1);
let current: ListNode = head;
while (list1 !== null && list2 !== null) {
if (list1.val < list2.val) {
current.next = list1;
list1 = list1.next;
} else {
current.next = list2;
list2 = list2.next;
}
current = current.next;
}
if (list1 === null) current.next = list2;
if (list2 === null) current.next = list1;
return head.next;
};
|
有效的括号
题目内容
解题思路
栈。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
| const isValid = (s: string): boolean => {
type KeyType = ")" | "}" | "]";
type ValueType = "(" | "{" | "[";
const n: number = s.length;
if (n % 2 === 1) return false;
const mapObj: Record<KeyType, ValueType> = {
")": "(",
"}": "{",
"]": "[",
};
const stk: ValueType[] = [];
for (let item of s) {
if (item in mapObj) {
let top = stk.pop();
if (mapObj[item] !== top) {
return false;
}
} else {
stk.push(item as ValueType);
}
}
return !stk.length;
};
|
两数之和
题目内容
解题思路
哈希表,两数之和变为两数之差。
代码实现
1
2
3
4
5
6
7
8
9
10
| const twoSum = (nums: number[], target: number): number[] => {
const mapObj: Map<number, number> = new Map();
for (let i = 0; i < nums.length; i++) {
if (mapObj.has(target - nums[i])) {
return [mapObj.get(target - nums[i]), i];
} else {
mapObj.set(nums[i], i);
}
}
};
|