mirror of
https://github.com/krahets/hello-algo.git
synced 2026-04-13 14:10:00 +08:00
build
This commit is contained in:
@@ -1443,8 +1443,8 @@ comments: true
|
||||
/* 双向链表节点类 */
|
||||
class ListNode {
|
||||
int val; // 节点值
|
||||
ListNode next; // 指向后继节点的引用
|
||||
ListNode prev; // 指向前驱节点的引用
|
||||
ListNode? next; // 指向后继节点的引用
|
||||
ListNode? prev; // 指向前驱节点的引用
|
||||
ListNode(this.val, [this.next, this.prev]); // 构造函数
|
||||
}
|
||||
```
|
||||
|
||||
@@ -1721,7 +1721,7 @@ comments: true
|
||||
remove(index) {
|
||||
if (index < 0 || index >= this.#size) throw new Error('索引越界');
|
||||
let num = this.#arr[index];
|
||||
// 将将索引 index 之后的元素都向前移动一位
|
||||
// 将索引 index 之后的元素都向前移动一位
|
||||
for (let j = index; j < this.#size - 1; j++) {
|
||||
this.#arr[j] = this.#arr[j + 1];
|
||||
}
|
||||
|
||||
@@ -29,7 +29,7 @@ comments: true
|
||||
|
||||
链表由节点组成,节点之间通过引用(指针)连接,各个节点可以存储不同类型的数据,例如 `int`、`double`、`string`、`object` 等。
|
||||
|
||||
相对地,数组元素则必须是相同类型的,这样才能通过计算偏移量来获取对应元素位置。例如,数组同时包含 `int` 和 `long` 两种类型,单个元素分别占用 4 字节 和 8 字节 ,此时就不能用以下公式计算偏移量了,因为数组中包含了两种“元素长度”。
|
||||
相对地,数组元素则必须是相同类型的,这样才能通过计算偏移量来获取对应元素位置。例如,数组同时包含 `int` 和 `long` 两种类型,单个元素分别占用 4 字节和 8 字节 ,此时就不能用以下公式计算偏移量了,因为数组中包含了两种“元素长度”。
|
||||
|
||||
```shell
|
||||
# 元素内存地址 = 数组内存地址(首元素内存地址) + 元素长度 * 元素索引
|
||||
|
||||
@@ -532,11 +532,7 @@ comments: true
|
||||
) {
|
||||
// 当放置完所有行时,记录解
|
||||
if row == n {
|
||||
let mut copy_state: Vec<Vec<String>> = Vec::new();
|
||||
for s_row in state.clone() {
|
||||
copy_state.push(s_row);
|
||||
}
|
||||
res.push(copy_state);
|
||||
res.push(state.clone());
|
||||
return;
|
||||
}
|
||||
// 遍历所有列
|
||||
@@ -547,12 +543,12 @@ comments: true
|
||||
// 剪枝:不允许该格子所在列、主对角线、次对角线上存在皇后
|
||||
if !cols[col] && !diags1[diag1] && !diags2[diag2] {
|
||||
// 尝试:将皇后放置在该格子
|
||||
state.get_mut(row).unwrap()[col] = "Q".into();
|
||||
state[row][col] = "Q".into();
|
||||
(cols[col], diags1[diag1], diags2[diag2]) = (true, true, true);
|
||||
// 放置下一行
|
||||
backtrack(row + 1, n, state, res, cols, diags1, diags2);
|
||||
// 回退:将该格子恢复为空位
|
||||
state.get_mut(row).unwrap()[col] = "#".into();
|
||||
state[row][col] = "#".into();
|
||||
(cols[col], diags1[diag1], diags2[diag2]) = (false, false, false);
|
||||
}
|
||||
}
|
||||
@@ -561,14 +557,7 @@ comments: true
|
||||
/* 求解 n 皇后 */
|
||||
fn n_queens(n: usize) -> Vec<Vec<Vec<String>>> {
|
||||
// 初始化 n*n 大小的棋盘,其中 'Q' 代表皇后,'#' 代表空位
|
||||
let mut state: Vec<Vec<String>> = Vec::new();
|
||||
for _ in 0..n {
|
||||
let mut row: Vec<String> = Vec::new();
|
||||
for _ in 0..n {
|
||||
row.push("#".into());
|
||||
}
|
||||
state.push(row);
|
||||
}
|
||||
let mut state: Vec<Vec<String>> = vec![vec!["#".to_string(); n]; n];
|
||||
let mut cols = vec![false; n]; // 记录列是否有皇后
|
||||
let mut diags1 = vec![false; 2 * n - 1]; // 记录主对角线上是否有皇后
|
||||
let mut diags2 = vec![false; 2 * n - 1]; // 记录次对角线上是否有皇后
|
||||
|
||||
@@ -355,7 +355,7 @@ comments: true
|
||||
```rust title="subset_sum_i_naive.rs"
|
||||
/* 回溯算法:子集和 I */
|
||||
fn backtrack(
|
||||
mut state: Vec<i32>,
|
||||
state: &mut Vec<i32>,
|
||||
target: i32,
|
||||
total: i32,
|
||||
choices: &[i32],
|
||||
@@ -363,7 +363,7 @@ comments: true
|
||||
) {
|
||||
// 子集和等于 target 时,记录解
|
||||
if total == target {
|
||||
res.push(state);
|
||||
res.push(state.clone());
|
||||
return;
|
||||
}
|
||||
// 遍历所有选择
|
||||
@@ -375,7 +375,7 @@ comments: true
|
||||
// 尝试:做出选择,更新元素和 total
|
||||
state.push(choices[i]);
|
||||
// 进行下一轮选择
|
||||
backtrack(state.clone(), target, total + choices[i], choices, res);
|
||||
backtrack(state, target, total + choices[i], choices, res);
|
||||
// 回退:撤销选择,恢复到之前的状态
|
||||
state.pop();
|
||||
}
|
||||
@@ -383,10 +383,10 @@ comments: true
|
||||
|
||||
/* 求解子集和 I(包含重复子集) */
|
||||
fn subset_sum_i_naive(nums: &[i32], target: i32) -> Vec<Vec<i32>> {
|
||||
let state = Vec::new(); // 状态(子集)
|
||||
let mut state = Vec::new(); // 状态(子集)
|
||||
let total = 0; // 子集和
|
||||
let mut res = Vec::new(); // 结果列表(子集列表)
|
||||
backtrack(state, target, total, nums, &mut res);
|
||||
backtrack(&mut state, target, total, nums, &mut res);
|
||||
res
|
||||
}
|
||||
```
|
||||
@@ -912,7 +912,7 @@ comments: true
|
||||
```rust title="subset_sum_i.rs"
|
||||
/* 回溯算法:子集和 I */
|
||||
fn backtrack(
|
||||
mut state: Vec<i32>,
|
||||
state: &mut Vec<i32>,
|
||||
target: i32,
|
||||
choices: &[i32],
|
||||
start: usize,
|
||||
@@ -920,7 +920,7 @@ comments: true
|
||||
) {
|
||||
// 子集和等于 target 时,记录解
|
||||
if target == 0 {
|
||||
res.push(state);
|
||||
res.push(state.clone());
|
||||
return;
|
||||
}
|
||||
// 遍历所有选择
|
||||
@@ -934,7 +934,7 @@ comments: true
|
||||
// 尝试:做出选择,更新 target, start
|
||||
state.push(choices[i]);
|
||||
// 进行下一轮选择
|
||||
backtrack(state.clone(), target - choices[i], choices, i, res);
|
||||
backtrack(state, target - choices[i], choices, i, res);
|
||||
// 回退:撤销选择,恢复到之前的状态
|
||||
state.pop();
|
||||
}
|
||||
@@ -942,11 +942,11 @@ comments: true
|
||||
|
||||
/* 求解子集和 I */
|
||||
fn subset_sum_i(nums: &mut [i32], target: i32) -> Vec<Vec<i32>> {
|
||||
let state = Vec::new(); // 状态(子集)
|
||||
let mut state = Vec::new(); // 状态(子集)
|
||||
nums.sort(); // 对 nums 进行排序
|
||||
let start = 0; // 遍历起始点
|
||||
let mut res = Vec::new(); // 结果列表(子集列表)
|
||||
backtrack(state, target, nums, start, &mut res);
|
||||
backtrack(&mut state, target, nums, start, &mut res);
|
||||
res
|
||||
}
|
||||
```
|
||||
@@ -1512,7 +1512,7 @@ comments: true
|
||||
```rust title="subset_sum_ii.rs"
|
||||
/* 回溯算法:子集和 II */
|
||||
fn backtrack(
|
||||
mut state: Vec<i32>,
|
||||
state: &mut Vec<i32>,
|
||||
target: i32,
|
||||
choices: &[i32],
|
||||
start: usize,
|
||||
@@ -1520,7 +1520,7 @@ comments: true
|
||||
) {
|
||||
// 子集和等于 target 时,记录解
|
||||
if target == 0 {
|
||||
res.push(state);
|
||||
res.push(state.clone());
|
||||
return;
|
||||
}
|
||||
// 遍历所有选择
|
||||
@@ -1539,7 +1539,7 @@ comments: true
|
||||
// 尝试:做出选择,更新 target, start
|
||||
state.push(choices[i]);
|
||||
// 进行下一轮选择
|
||||
backtrack(state.clone(), target - choices[i], choices, i, res);
|
||||
backtrack(state, target - choices[i], choices, i + 1, res);
|
||||
// 回退:撤销选择,恢复到之前的状态
|
||||
state.pop();
|
||||
}
|
||||
@@ -1547,11 +1547,11 @@ comments: true
|
||||
|
||||
/* 求解子集和 II */
|
||||
fn subset_sum_ii(nums: &mut [i32], target: i32) -> Vec<Vec<i32>> {
|
||||
let state = Vec::new(); // 状态(子集)
|
||||
let mut state = Vec::new(); // 状态(子集)
|
||||
nums.sort(); // 对 nums 进行排序
|
||||
let start = 0; // 遍历起始点
|
||||
let mut res = Vec::new(); // 结果列表(子集列表)
|
||||
backtrack(state, target, nums, start, &mut res);
|
||||
backtrack(&mut state, target, nums, start, &mut res);
|
||||
res
|
||||
}
|
||||
```
|
||||
|
||||
@@ -54,7 +54,7 @@ $$
|
||||
|
||||
1. 初始状态下,指针 $i$ 和 $j$ 分列数组两端。
|
||||
2. 计算当前状态的容量 $cap[i, j]$ ,并更新最大容量。
|
||||
3. 比较板 $i$ 和 板 $j$ 的高度,并将短板向内移动一格。
|
||||
3. 比较板 $i$ 和板 $j$ 的高度,并将短板向内移动一格。
|
||||
4. 循环执行第 `2.` 步和第 `3.` 步,直至 $i$ 和 $j$ 相遇时结束。
|
||||
|
||||
=== "<1>"
|
||||
|
||||
@@ -12,7 +12,7 @@ comments: true
|
||||
|
||||
1. 翻开字典约一半的页数,查看该页的首字母是什么,假设首字母为 $m$ 。
|
||||
2. 由于在拼音字母表中 $r$ 位于 $m$ 之后,所以排除字典前半部分,查找范围缩小到后半部分。
|
||||
3. 不断重复步骤 `1.` 和 步骤 `2.` ,直至找到拼音首字母为 $r$ 的页码为止。
|
||||
3. 不断重复步骤 `1.` 和步骤 `2.` ,直至找到拼音首字母为 $r$ 的页码为止。
|
||||
|
||||
=== "<1>"
|
||||
{ class="animation-figure" }
|
||||
|
||||
@@ -69,26 +69,19 @@ comments: true
|
||||
=== "C++"
|
||||
|
||||
```cpp title="quick_sort.cpp"
|
||||
/* 元素交换 */
|
||||
void swap(vector<int> &nums, int i, int j) {
|
||||
int tmp = nums[i];
|
||||
nums[i] = nums[j];
|
||||
nums[j] = tmp;
|
||||
}
|
||||
|
||||
/* 哨兵划分 */
|
||||
int partition(vector<int> &nums, int left, int right) {
|
||||
// 以 nums[left] 为基准数
|
||||
int i = left, j = right;
|
||||
while (i < j) {
|
||||
while (i < j && nums[j] >= nums[left])
|
||||
j--; // 从右向左找首个小于基准数的元素
|
||||
j--; // 从右向左找首个小于基准数的元素
|
||||
while (i < j && nums[i] <= nums[left])
|
||||
i++; // 从左向右找首个大于基准数的元素
|
||||
swap(nums, i, j); // 交换这两个元素
|
||||
i++; // 从左向右找首个大于基准数的元素
|
||||
swap(nums[i], nums[j]); // 交换这两个元素
|
||||
}
|
||||
swap(nums, i, left); // 将基准数交换至两子数组的分界线
|
||||
return i; // 返回基准数的索引
|
||||
swap(nums[i], nums[left]); // 将基准数交换至两子数组的分界线
|
||||
return i; // 返回基准数的索引
|
||||
}
|
||||
```
|
||||
|
||||
@@ -721,18 +714,18 @@ comments: true
|
||||
// 选取三个候选元素的中位数
|
||||
int med = medianThree(nums, left, (left + right) / 2, right);
|
||||
// 将中位数交换至数组最左端
|
||||
swap(nums, left, med);
|
||||
swap(nums[left], nums[med]);
|
||||
// 以 nums[left] 为基准数
|
||||
int i = left, j = right;
|
||||
while (i < j) {
|
||||
while (i < j && nums[j] >= nums[left])
|
||||
j--; // 从右向左找首个小于基准数的元素
|
||||
j--; // 从右向左找首个小于基准数的元素
|
||||
while (i < j && nums[i] <= nums[left])
|
||||
i++; // 从左向右找首个大于基准数的元素
|
||||
swap(nums, i, j); // 交换这两个元素
|
||||
i++; // 从左向右找首个大于基准数的元素
|
||||
swap(nums[i], nums[j]); // 交换这两个元素
|
||||
}
|
||||
swap(nums, i, left); // 将基准数交换至两子数组的分界线
|
||||
return i; // 返回基准数的索引
|
||||
swap(nums[i], nums[left]); // 将基准数交换至两子数组的分界线
|
||||
return i; // 返回基准数的索引
|
||||
}
|
||||
```
|
||||
|
||||
|
||||
@@ -665,7 +665,7 @@ comments: true
|
||||
|
||||
### 2. 完全二叉树
|
||||
|
||||
如图 7-5 所示,<u>完全二叉树(complete binary tree)</u>只有最底层的节点未被填满,且最底层节点尽量靠左填充。
|
||||
如图 7-5 所示,<u>完全二叉树(complete binary tree)</u>只有最底层的节点未被填满,且最底层节点尽量靠左填充。请注意,完美二叉树也是一棵完全二叉树。
|
||||
|
||||
{ class="animation-figure" }
|
||||
|
||||
|
||||
Reference in New Issue
Block a user