mirror of
https://github.com/krahets/hello-algo.git
synced 2025-11-05 06:37:12 +08:00
Add missing Dart codes and fix some errors (#689)
* Add missing Dart codes and fix some errors * Update array_binary_tree.dart --------- Co-authored-by: Yudong Jin <krahets@163.com>
This commit is contained in:
@ -20,8 +20,8 @@ void constant(int n) {
|
|||||||
// 常量、变量、对象占用 O(1) 空间
|
// 常量、变量、对象占用 O(1) 空间
|
||||||
final int a = 0;
|
final int a = 0;
|
||||||
int b = 0;
|
int b = 0;
|
||||||
|
|
||||||
List<int> nums = List.filled(10000, 0);
|
List<int> nums = List.filled(10000, 0);
|
||||||
|
ListNode node = ListNode(0);
|
||||||
// 循环中的变量占用 O(1) 空间
|
// 循环中的变量占用 O(1) 空间
|
||||||
for (var i = 0; i < n; i++) {
|
for (var i = 0; i < n; i++) {
|
||||||
int c = 0;
|
int c = 0;
|
||||||
@ -61,7 +61,6 @@ void quadratic(int n) {
|
|||||||
List<List<int>> numMatrix = List.generate(n, (_) => List.filled(n, 0));
|
List<List<int>> numMatrix = List.generate(n, (_) => List.filled(n, 0));
|
||||||
// 二维列表占用 O(n^2) 空间
|
// 二维列表占用 O(n^2) 空间
|
||||||
List<List<int>> numList = [];
|
List<List<int>> numList = [];
|
||||||
|
|
||||||
for (var i = 0; i < n; i++) {
|
for (var i = 0; i < n; i++) {
|
||||||
List<int> tmp = [];
|
List<int> tmp = [];
|
||||||
for (int j = 0; j < n; j++) {
|
for (int j = 0; j < n; j++) {
|
||||||
|
|||||||
@ -6,6 +6,7 @@
|
|||||||
|
|
||||||
import '../utils/print_util.dart';
|
import '../utils/print_util.dart';
|
||||||
|
|
||||||
|
/* 大顶堆 */
|
||||||
class MaxHeap {
|
class MaxHeap {
|
||||||
late List<int> _maxHeap;
|
late List<int> _maxHeap;
|
||||||
|
|
||||||
@ -15,7 +16,7 @@ class MaxHeap {
|
|||||||
_maxHeap = nums;
|
_maxHeap = nums;
|
||||||
// 堆化除叶节点以外的其他所有节点
|
// 堆化除叶节点以外的其他所有节点
|
||||||
for (int i = _parent(size() - 1); i >= 0; i--) {
|
for (int i = _parent(size() - 1); i >= 0; i--) {
|
||||||
_siftDown(i);
|
siftDown(i);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -61,11 +62,11 @@ class MaxHeap {
|
|||||||
// 添加节点
|
// 添加节点
|
||||||
_maxHeap.add(val);
|
_maxHeap.add(val);
|
||||||
// 从底至顶堆化
|
// 从底至顶堆化
|
||||||
_siftUp(size() - 1);
|
siftUp(size() - 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 从节点 i 开始,从底至顶堆化 */
|
/* 从节点 i 开始,从底至顶堆化 */
|
||||||
void _siftUp(int i) {
|
void siftUp(int i) {
|
||||||
while (true) {
|
while (true) {
|
||||||
// 获取节点 i 的父节点
|
// 获取节点 i 的父节点
|
||||||
int p = _parent(i);
|
int p = _parent(i);
|
||||||
@ -89,13 +90,13 @@ class MaxHeap {
|
|||||||
// 删除节点
|
// 删除节点
|
||||||
int val = _maxHeap.removeLast();
|
int val = _maxHeap.removeLast();
|
||||||
// 从顶至底堆化
|
// 从顶至底堆化
|
||||||
_siftDown(0);
|
siftDown(0);
|
||||||
// 返回堆顶元素
|
// 返回堆顶元素
|
||||||
return val;
|
return val;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 从节点 i 开始,从顶至底堆化 */
|
/* 从节点 i 开始,从顶至底堆化 */
|
||||||
void _siftDown(int i) {
|
void siftDown(int i) {
|
||||||
while (true) {
|
while (true) {
|
||||||
// 判断节点 i, l, r 中值最大的节点,记为 ma
|
// 判断节点 i, l, r 中值最大的节点,记为 ma
|
||||||
int l = _left(i);
|
int l = _left(i);
|
||||||
|
|||||||
150
codes/dart/chapter_heap/top_k.dart
Normal file
150
codes/dart/chapter_heap/top_k.dart
Normal file
@ -0,0 +1,150 @@
|
|||||||
|
/**
|
||||||
|
* File: top_k.dart
|
||||||
|
* Created Time: 2023-08-15
|
||||||
|
* Author: liuyuxin (gvenusleo@gmail.com)
|
||||||
|
*/
|
||||||
|
|
||||||
|
import '../utils/print_util.dart';
|
||||||
|
|
||||||
|
/* 基于堆查找数组中最大的 k 个元素 */
|
||||||
|
MinHeap topKHeap(List<int> nums, int k) {
|
||||||
|
// 将数组的前 k 个元素入堆
|
||||||
|
MinHeap heap = MinHeap(nums.sublist(0, k));
|
||||||
|
// 从第 k+1 个元素开始,保持堆的长度为 k
|
||||||
|
for (int i = k; i < nums.length; i++) {
|
||||||
|
// 若当前元素大于堆顶元素,则将堆顶元素出堆、当前元素入堆
|
||||||
|
if (nums[i] > heap.peek()) {
|
||||||
|
heap.pop();
|
||||||
|
heap.push(nums[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return heap;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Driver Code */
|
||||||
|
void main() {
|
||||||
|
List<int> nums = [1, 7, 6, 3, 2];
|
||||||
|
int k = 3;
|
||||||
|
|
||||||
|
MinHeap res = topKHeap(nums, k);
|
||||||
|
print("最大的 $k 个元素为");
|
||||||
|
res.print();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 小顶堆 */
|
||||||
|
class MinHeap {
|
||||||
|
late List<int> _minHeap;
|
||||||
|
|
||||||
|
/* 构造方法,根据输入列表建堆 */
|
||||||
|
MinHeap(List<int> nums) {
|
||||||
|
// 将列表元素原封不动添加进堆
|
||||||
|
_minHeap = nums;
|
||||||
|
// 堆化除叶节点以外的其他所有节点
|
||||||
|
for (int i = _parent(size() - 1); i >= 0; i--) {
|
||||||
|
siftDown(i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 返回堆中的元素 */
|
||||||
|
List<int> getHeap() {
|
||||||
|
return _minHeap;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 获取左子节点索引 */
|
||||||
|
int _left(int i) {
|
||||||
|
return 2 * i + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 获取右子节点索引 */
|
||||||
|
int _right(int i) {
|
||||||
|
return 2 * i + 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 获取父节点索引 */
|
||||||
|
int _parent(int i) {
|
||||||
|
return (i - 1) ~/ 2; // 向下整除
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 交换元素 */
|
||||||
|
void _swap(int i, int j) {
|
||||||
|
int tmp = _minHeap[i];
|
||||||
|
_minHeap[i] = _minHeap[j];
|
||||||
|
_minHeap[j] = tmp;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 获取堆大小 */
|
||||||
|
int size() {
|
||||||
|
return _minHeap.length;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 判断堆是否为空 */
|
||||||
|
bool isEmpty() {
|
||||||
|
return size() == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 访问堆顶元素 */
|
||||||
|
int peek() {
|
||||||
|
return _minHeap[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 元素入堆 */
|
||||||
|
void push(int val) {
|
||||||
|
// 添加节点
|
||||||
|
_minHeap.add(val);
|
||||||
|
// 从底至顶堆化
|
||||||
|
siftUp(size() - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 从节点 i 开始,从底至顶堆化 */
|
||||||
|
void siftUp(int i) {
|
||||||
|
while (true) {
|
||||||
|
// 获取节点 i 的父节点
|
||||||
|
int p = _parent(i);
|
||||||
|
// 当“越过根节点”或“节点无需修复”时,结束堆化
|
||||||
|
if (p < 0 || _minHeap[i] >= _minHeap[p]) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
// 交换两节点
|
||||||
|
_swap(i, p);
|
||||||
|
// 循环向上堆化
|
||||||
|
i = p;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 元素出堆 */
|
||||||
|
int pop() {
|
||||||
|
// 判空处理
|
||||||
|
if (isEmpty()) throw Exception('堆为空');
|
||||||
|
// 交换根节点与最右叶节点(即交换首元素与尾元素)
|
||||||
|
_swap(0, size() - 1);
|
||||||
|
// 删除节点
|
||||||
|
int val = _minHeap.removeLast();
|
||||||
|
// 从顶至底堆化
|
||||||
|
siftDown(0);
|
||||||
|
// 返回堆顶元素
|
||||||
|
return val;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 从节点 i 开始,从顶至底堆化 */
|
||||||
|
void siftDown(int i) {
|
||||||
|
while (true) {
|
||||||
|
// 判断节点 i, l, r 中值最大的节点,记为 ma
|
||||||
|
int l = _left(i);
|
||||||
|
int r = _right(i);
|
||||||
|
int mi = i;
|
||||||
|
if (l < size() && _minHeap[l] < _minHeap[mi]) mi = l;
|
||||||
|
if (r < size() && _minHeap[r] < _minHeap[mi]) mi = r;
|
||||||
|
// 若节点 i 最大或索引 l, r 越界,则无需继续堆化,跳出
|
||||||
|
if (mi == i) break;
|
||||||
|
// 交换两节点
|
||||||
|
_swap(i, mi);
|
||||||
|
// 循环向下堆化
|
||||||
|
i = mi;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 打印堆(二叉树) */
|
||||||
|
void print() {
|
||||||
|
printHeap(_minHeap);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -9,6 +9,7 @@ import 'dart:collection';
|
|||||||
/* 方法一: 暴力枚举 */
|
/* 方法一: 暴力枚举 */
|
||||||
List<int> twoSumBruteForce(List<int> nums, int target) {
|
List<int> twoSumBruteForce(List<int> nums, int target) {
|
||||||
int size = nums.length;
|
int size = nums.length;
|
||||||
|
// 两层循环,时间复杂度 O(n^2)
|
||||||
for (var i = 0; i < size - 1; i++) {
|
for (var i = 0; i < size - 1; i++) {
|
||||||
for (var j = i + 1; j < size; j++) {
|
for (var j = i + 1; j < size; j++) {
|
||||||
if (nums[i] + nums[j] == target) return [i, j];
|
if (nums[i] + nums[j] == target) return [i, j];
|
||||||
@ -20,7 +21,9 @@ List<int> twoSumBruteForce(List<int> nums, int target) {
|
|||||||
/* 方法二: 辅助哈希表 */
|
/* 方法二: 辅助哈希表 */
|
||||||
List<int> twoSumHashTable(List<int> nums, int target) {
|
List<int> twoSumHashTable(List<int> nums, int target) {
|
||||||
int size = nums.length;
|
int size = nums.length;
|
||||||
|
// 辅助哈希表,空间复杂度 O(n)
|
||||||
Map<int, int> dic = HashMap();
|
Map<int, int> dic = HashMap();
|
||||||
|
// 单层循环,时间复杂度 O(n)
|
||||||
for (var i = 0; i < size; i++) {
|
for (var i = 0; i < size; i++) {
|
||||||
if (dic.containsKey(target - nums[i])) {
|
if (dic.containsKey(target - nums[i])) {
|
||||||
return [dic[target - nums[i]]!, i];
|
return [dic[target - nums[i]]!, i];
|
||||||
|
|||||||
152
codes/dart/chapter_tree/array_binary_tree.dart
Normal file
152
codes/dart/chapter_tree/array_binary_tree.dart
Normal file
@ -0,0 +1,152 @@
|
|||||||
|
/**
|
||||||
|
* File: array_binary_tree.dart
|
||||||
|
* Created Time: 2023-08-15
|
||||||
|
* Author: liuyuxin (gvenusleo@gmail.com)
|
||||||
|
*/
|
||||||
|
|
||||||
|
import '../utils/print_util.dart';
|
||||||
|
import '../utils/tree_node.dart';
|
||||||
|
|
||||||
|
/* 数组表示下的二叉树类 */
|
||||||
|
class ArrayBinaryTree {
|
||||||
|
late List<int?> _tree;
|
||||||
|
|
||||||
|
/* 构造方法 */
|
||||||
|
ArrayBinaryTree(this._tree);
|
||||||
|
|
||||||
|
/* 节点数量 */
|
||||||
|
int size() {
|
||||||
|
return _tree.length;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 获取索引为 i 节点的值 */
|
||||||
|
int? val(int i) {
|
||||||
|
// 若索引越界,则返回 null ,代表空位
|
||||||
|
if (i < 0 || i >= size()) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return _tree[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 获取索引为 i 节点的左子节点的索引 */
|
||||||
|
int? left(int i) {
|
||||||
|
return 2 * i + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 获取索引为 i 节点的右子节点的索引 */
|
||||||
|
int? right(int i) {
|
||||||
|
return 2 * i + 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 获取索引为 i 节点的父节点的索引 */
|
||||||
|
int? parent(int i) {
|
||||||
|
return (i - 1) ~/ 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 层序遍历 */
|
||||||
|
List<int> levelOrder() {
|
||||||
|
List<int> res = [];
|
||||||
|
for (int i = 0; i < size(); i++) {
|
||||||
|
if (val(i) != null) {
|
||||||
|
res.add(val(i)!);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 深度优先遍历 */
|
||||||
|
void dfs(int i, String order, List<int?> res) {
|
||||||
|
// 若为空位,则返回
|
||||||
|
if (val(i) == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// 前序遍历
|
||||||
|
if (order == 'pre') {
|
||||||
|
res.add(val(i));
|
||||||
|
}
|
||||||
|
dfs(left(i)!, order, res);
|
||||||
|
// 中序遍历
|
||||||
|
if (order == 'in') {
|
||||||
|
res.add(val(i));
|
||||||
|
}
|
||||||
|
dfs(right(i)!, order, res);
|
||||||
|
// 后序遍历
|
||||||
|
if (order == 'post') {
|
||||||
|
res.add(val(i));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 前序遍历 */
|
||||||
|
List<int?> preOrder() {
|
||||||
|
List<int?> res = [];
|
||||||
|
dfs(0, 'pre', res);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 中序遍历 */
|
||||||
|
List<int?> inOrder() {
|
||||||
|
List<int?> res = [];
|
||||||
|
dfs(0, 'in', res);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 后序遍历 */
|
||||||
|
List<int?> postOrder() {
|
||||||
|
List<int?> res = [];
|
||||||
|
dfs(0, 'post', res);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Driver Code */
|
||||||
|
void main() {
|
||||||
|
// 初始化二叉树
|
||||||
|
// 这里借助了一个从数组直接生成二叉树的函数
|
||||||
|
List<int?> arr = [
|
||||||
|
1,
|
||||||
|
2,
|
||||||
|
3,
|
||||||
|
4,
|
||||||
|
null,
|
||||||
|
6,
|
||||||
|
7,
|
||||||
|
8,
|
||||||
|
9,
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
12,
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
15
|
||||||
|
];
|
||||||
|
|
||||||
|
TreeNode? root = listToTree(arr);
|
||||||
|
print("\n初始化二叉树\n");
|
||||||
|
print("二叉树的数组表示:");
|
||||||
|
print(arr);
|
||||||
|
print("二叉树的链表表示:");
|
||||||
|
printTree(root);
|
||||||
|
|
||||||
|
// 数组表示下的二叉树类
|
||||||
|
ArrayBinaryTree abt = ArrayBinaryTree(arr);
|
||||||
|
|
||||||
|
// 访问节点
|
||||||
|
int i = 1;
|
||||||
|
int? l = abt.left(i);
|
||||||
|
int? r = abt.right(i);
|
||||||
|
int? p = abt.parent(i);
|
||||||
|
print("\n当前节点的索引为 $i ,值为 ${abt.val(i)}");
|
||||||
|
print("其左子节点的索引为 $l ,值为 ${(l == null ? "null" : abt.val(l))}");
|
||||||
|
print("其右子节点的索引为 $r ,值为 ${(r == null ? "null" : abt.val(r))}");
|
||||||
|
print("其父节点的索引为 $p ,值为 ${(p == null ? "null" : abt.val(p))}");
|
||||||
|
|
||||||
|
// 遍历树
|
||||||
|
List<int?> res = abt.levelOrder();
|
||||||
|
print("\n层序遍历为:$res");
|
||||||
|
res = abt.preOrder();
|
||||||
|
print("前序遍历为 $res");
|
||||||
|
res = abt.inOrder();
|
||||||
|
print("中序遍历为 $res");
|
||||||
|
res = abt.postOrder();
|
||||||
|
print("后序遍历为 $res");
|
||||||
|
}
|
||||||
@ -18,6 +18,7 @@ class AVLTree {
|
|||||||
|
|
||||||
/* 获取节点高度 */
|
/* 获取节点高度 */
|
||||||
int height(TreeNode? node) {
|
int height(TreeNode? node) {
|
||||||
|
// 空节点高度为 -1 ,叶节点高度为 0
|
||||||
return node == null ? -1 : node.height;
|
return node == null ? -1 : node.height;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -8,20 +8,22 @@ import '../utils/print_util.dart';
|
|||||||
import '../utils/tree_node.dart';
|
import '../utils/tree_node.dart';
|
||||||
|
|
||||||
/* 二叉搜索树 */
|
/* 二叉搜索树 */
|
||||||
TreeNode? root;
|
class BinarySearchTree {
|
||||||
|
late TreeNode? _root;
|
||||||
|
|
||||||
void binarySearchTree(List<int> nums) {
|
/* 构造方法 */
|
||||||
|
BinarySearchTree(List<int> nums) {
|
||||||
nums.sort(); // 排序数组
|
nums.sort(); // 排序数组
|
||||||
root = buildTree(nums, 0, nums.length - 1); // 构建二叉搜索树
|
_root = buildTree(nums, 0, nums.length - 1); // 构建二叉搜索树
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 获取二叉树的根节点 */
|
/* 获取二叉树的根节点 */
|
||||||
TreeNode? getRoot() {
|
TreeNode? getRoot() {
|
||||||
return root;
|
return _root;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 构建二叉上搜索树 */
|
/* 构建二叉上搜索树 */
|
||||||
TreeNode? buildTree(List<int> nums, int i, int j) {
|
TreeNode? buildTree(List<int> nums, int i, int j) {
|
||||||
if (i > j) {
|
if (i > j) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@ -31,11 +33,11 @@ TreeNode? buildTree(List<int> nums, int i, int j) {
|
|||||||
root.left = buildTree(nums, i, mid - 1);
|
root.left = buildTree(nums, i, mid - 1);
|
||||||
root.right = buildTree(nums, mid + 1, j);
|
root.right = buildTree(nums, mid + 1, j);
|
||||||
return root;
|
return root;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 查找节点 */
|
/* 查找节点 */
|
||||||
TreeNode? search(int num) {
|
TreeNode? search(int num) {
|
||||||
TreeNode? cur = root;
|
TreeNode? cur = _root;
|
||||||
// 循环查找,越过叶节点后跳出
|
// 循环查找,越过叶节点后跳出
|
||||||
while (cur != null) {
|
while (cur != null) {
|
||||||
// 目标节点在 cur 的右子树中
|
// 目标节点在 cur 的右子树中
|
||||||
@ -50,13 +52,13 @@ TreeNode? search(int num) {
|
|||||||
}
|
}
|
||||||
// 返回目标节点
|
// 返回目标节点
|
||||||
return cur;
|
return cur;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 插入节点 */
|
/* 插入节点 */
|
||||||
void insert(int num) {
|
void insert(int num) {
|
||||||
// 若树为空,直接提前返回
|
// 若树为空,直接提前返回
|
||||||
if (root == null) return;
|
if (_root == null) return;
|
||||||
TreeNode? cur = root;
|
TreeNode? cur = _root;
|
||||||
TreeNode? pre = null;
|
TreeNode? pre = null;
|
||||||
// 循环查找,越过叶节点后跳出
|
// 循环查找,越过叶节点后跳出
|
||||||
while (cur != null) {
|
while (cur != null) {
|
||||||
@ -76,14 +78,14 @@ void insert(int num) {
|
|||||||
pre.right = node;
|
pre.right = node;
|
||||||
else
|
else
|
||||||
pre.left = node;
|
pre.left = node;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 删除节点 */
|
/* 删除节点 */
|
||||||
void remove(int num) {
|
void remove(int num) {
|
||||||
// 若树为空,直接提前返回
|
// 若树为空,直接提前返回
|
||||||
if (root == null) return;
|
if (_root == null) return;
|
||||||
|
|
||||||
TreeNode? cur = root;
|
TreeNode? cur = _root;
|
||||||
TreeNode? pre = null;
|
TreeNode? pre = null;
|
||||||
// 循环查找,越过叶节点后跳出
|
// 循环查找,越过叶节点后跳出
|
||||||
while (cur != null) {
|
while (cur != null) {
|
||||||
@ -104,14 +106,14 @@ void remove(int num) {
|
|||||||
// 当子节点数量 = 0 / 1 时, child = null / 该子节点
|
// 当子节点数量 = 0 / 1 时, child = null / 该子节点
|
||||||
TreeNode? child = cur.left ?? cur.right;
|
TreeNode? child = cur.left ?? cur.right;
|
||||||
// 删除节点 cur
|
// 删除节点 cur
|
||||||
if (cur != root) {
|
if (cur != _root) {
|
||||||
if (pre!.left == cur)
|
if (pre!.left == cur)
|
||||||
pre.left = child;
|
pre.left = child;
|
||||||
else
|
else
|
||||||
pre.right = child;
|
pre.right = child;
|
||||||
} else {
|
} else {
|
||||||
// 若删除节点为根节点,则重新指定根节点
|
// 若删除节点为根节点,则重新指定根节点
|
||||||
root = child;
|
_root = child;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// 子节点数量 = 2
|
// 子节点数量 = 2
|
||||||
@ -125,33 +127,34 @@ void remove(int num) {
|
|||||||
// 用 tmp 覆盖 cur
|
// 用 tmp 覆盖 cur
|
||||||
cur.val = tmp.val;
|
cur.val = tmp.val;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Driver Code */
|
/* Driver Code */
|
||||||
void main() {
|
void main() {
|
||||||
/* 初始化二叉搜索树 */
|
/* 初始化二叉搜索树 */
|
||||||
List<int> nums = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15];
|
List<int> nums = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15];
|
||||||
binarySearchTree(nums);
|
BinarySearchTree bst = BinarySearchTree(nums);
|
||||||
print("\n初始化的二叉树为\n");
|
print("\n初始化的二叉树为\n");
|
||||||
printTree(getRoot());
|
printTree(bst.getRoot());
|
||||||
|
|
||||||
/* 查找节点 */
|
/* 查找节点 */
|
||||||
TreeNode? node = search(7);
|
TreeNode? node = bst.search(7);
|
||||||
print("\n查找到的节点对象为 $node,节点值 = ${node?.val}");
|
print("\n查找到的节点对象为 $node,节点值 = ${node?.val}");
|
||||||
|
|
||||||
/* 插入节点 */
|
/* 插入节点 */
|
||||||
insert(16);
|
bst.insert(16);
|
||||||
print("\n插入节点 16 后,二叉树为\n");
|
print("\n插入节点 16 后,二叉树为\n");
|
||||||
printTree(getRoot());
|
printTree(bst.getRoot());
|
||||||
|
|
||||||
/* 删除节点 */
|
/* 删除节点 */
|
||||||
remove(1);
|
bst.remove(1);
|
||||||
print("\n删除节点 1 后,二叉树为\n");
|
print("\n删除节点 1 后,二叉树为\n");
|
||||||
printTree(getRoot());
|
printTree(bst.getRoot());
|
||||||
remove(2);
|
bst.remove(2);
|
||||||
print("\n删除节点 2 后,二叉树为\n");
|
print("\n删除节点 2 后,二叉树为\n");
|
||||||
printTree(getRoot());
|
printTree(bst.getRoot());
|
||||||
remove(4);
|
bst.remove(4);
|
||||||
print("\n删除节点 4 后,二叉树为\n");
|
print("\n删除节点 4 后,二叉树为\n");
|
||||||
printTree(getRoot());
|
printTree(bst.getRoot());
|
||||||
}
|
}
|
||||||
|
|||||||
@ -4,19 +4,12 @@
|
|||||||
* Author: Jefferson (JeffersonHuang77@gmail.com)
|
* Author: Jefferson (JeffersonHuang77@gmail.com)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/* Definition for a singly-linked list node */
|
||||||
* Definition for a singly-linked list node
|
|
||||||
*/
|
|
||||||
class ListNode {
|
class ListNode {
|
||||||
int val;
|
int val;
|
||||||
ListNode? next;
|
ListNode? next;
|
||||||
|
|
||||||
ListNode(this.val, [this.next]);
|
ListNode(this.val, [this.next]);
|
||||||
|
|
||||||
@override
|
|
||||||
String toString() {
|
|
||||||
return 'ListNode{val: $val, next: $next}';
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Generate a linked list with a vector */
|
/* Generate a linked list with a vector */
|
||||||
|
|||||||
@ -16,6 +16,7 @@ class Trunk {
|
|||||||
Trunk(this.prev, this.str);
|
Trunk(this.prev, this.str);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Print a matrix (Array) */
|
||||||
void printMatrix(List<List<int>> matrix) {
|
void printMatrix(List<List<int>> matrix) {
|
||||||
print("[");
|
print("[");
|
||||||
for (List<int> row in matrix) {
|
for (List<int> row in matrix) {
|
||||||
@ -24,6 +25,7 @@ void printMatrix(List<List<int>> matrix) {
|
|||||||
print("]");
|
print("]");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Print a linked list */
|
||||||
void printLinkedList(ListNode? head) {
|
void printLinkedList(ListNode? head) {
|
||||||
List<String> list = [];
|
List<String> list = [];
|
||||||
|
|
||||||
@ -35,6 +37,11 @@ void printLinkedList(ListNode? head) {
|
|||||||
print(list.join(' -> '));
|
print(list.join(' -> '));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The interface of the tree printer
|
||||||
|
* This tree printer is borrowed from TECHIE DELIGHT
|
||||||
|
* https://www.techiedelight.com/c-program-print-binary-tree/
|
||||||
|
*/
|
||||||
void printTree(TreeNode? root, [Trunk? prev = null, bool isLeft = false]) {
|
void printTree(TreeNode? root, [Trunk? prev = null, bool isLeft = false]) {
|
||||||
if (root == null) {
|
if (root == null) {
|
||||||
return;
|
return;
|
||||||
@ -65,6 +72,7 @@ void printTree(TreeNode? root, [Trunk? prev = null, bool isLeft = false]) {
|
|||||||
printTree(root.left, trunk, false);
|
printTree(root.left, trunk, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Helper function to print branches of the binary tree */
|
||||||
void showTrunks(Trunk? p) {
|
void showTrunks(Trunk? p) {
|
||||||
if (p == null) {
|
if (p == null) {
|
||||||
return;
|
return;
|
||||||
@ -74,6 +82,7 @@ void showTrunks(Trunk? p) {
|
|||||||
stdout.write(p.str);
|
stdout.write(p.str);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Print a heap (PriorityQueue) */
|
||||||
void printHeap(List<int> heap) {
|
void printHeap(List<int> heap) {
|
||||||
print("堆的数组表示:$heap");
|
print("堆的数组表示:$heap");
|
||||||
print("堆的树状表示:");
|
print("堆的树状表示:");
|
||||||
|
|||||||
@ -4,63 +4,47 @@
|
|||||||
* Author: Jefferson (JeffersonHuang77@gmail.com)
|
* Author: Jefferson (JeffersonHuang77@gmail.com)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import 'dart:collection';
|
/* 二叉树节点类 */
|
||||||
|
|
||||||
class TreeNode {
|
class TreeNode {
|
||||||
int val; // 节点值
|
int val; // 节点值
|
||||||
int height; // 节点高度
|
int height; // 节点高度
|
||||||
TreeNode? left; // 左子节点引用
|
TreeNode? left; // 左子节点引用
|
||||||
TreeNode? right; // 右子节点引用
|
TreeNode? right; // 右子节点引用
|
||||||
|
|
||||||
|
/* 构造方法 */
|
||||||
TreeNode(this.val, [this.height = 0, this.left, this.right]);
|
TreeNode(this.val, [this.height = 0, this.left, this.right]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/* 将列表反序列化为二叉树:递归 */
|
||||||
* Generate a binary tree given an array
|
TreeNode? listToTreeDFS(List<int?> arr, int i) {
|
||||||
* @param list
|
if (i < 0 || i >= arr.length || arr[i] == null) {
|
||||||
* @return
|
return null;
|
||||||
*/
|
|
||||||
TreeNode? listToTree(List<int> list) {
|
|
||||||
int size = list.length;
|
|
||||||
if (size == 0) return null;
|
|
||||||
|
|
||||||
TreeNode root = TreeNode(list[0]);
|
|
||||||
Queue<TreeNode?> queue = Queue();
|
|
||||||
queue.add(root);
|
|
||||||
int i = 0;
|
|
||||||
while (queue.isNotEmpty) {
|
|
||||||
TreeNode? node = queue.removeFirst();
|
|
||||||
if (++i >= size) break;
|
|
||||||
node?.left = TreeNode(list[i]);
|
|
||||||
queue.add(node?.left);
|
|
||||||
if (++i >= size) break;
|
|
||||||
node?.right = TreeNode(list[i]);
|
|
||||||
queue.add(node?.right);
|
|
||||||
}
|
}
|
||||||
|
TreeNode? root = TreeNode(arr[i]!);
|
||||||
|
root.left = listToTreeDFS(arr, 2 * i + 1);
|
||||||
|
root.right = listToTreeDFS(arr, 2 * i + 2);
|
||||||
return root;
|
return root;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/* 将列表反序列化为二叉树 */
|
||||||
* Serialize a binary tree to a list
|
TreeNode? listToTree(List<int?> arr) {
|
||||||
* @param root
|
return listToTreeDFS(arr, 0);
|
||||||
* @return
|
}
|
||||||
*/
|
|
||||||
List<int?> treeToList(TreeNode? root) {
|
/* 将二叉树序列化为列表:递归 */
|
||||||
List<int?> list = [];
|
void treeToListDFS(TreeNode? root, int i, List<int?> res) {
|
||||||
if (root == null) return list;
|
if (root == null) return;
|
||||||
Queue<TreeNode?> queue = Queue();
|
while (i >= res.length) {
|
||||||
queue.add(root);
|
res.add(null);
|
||||||
|
}
|
||||||
while (!queue.isEmpty) {
|
res[i] = root.val;
|
||||||
TreeNode? node = queue.first;
|
treeToListDFS(root.left, 2 * i + 1, res);
|
||||||
queue.removeFirst();
|
treeToListDFS(root.right, 2 * i + 2, res);
|
||||||
if (node != null) {
|
}
|
||||||
list.add(node.val);
|
|
||||||
queue.add(node.left);
|
/* 将二叉树序列化为列表 */
|
||||||
queue.add(node.right);
|
List<int?> treeToList(TreeNode? root) {
|
||||||
} else {
|
List<int?> res = [];
|
||||||
list.add(null);
|
treeToListDFS(root, 0, res);
|
||||||
}
|
return res;
|
||||||
}
|
|
||||||
return list;
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -44,6 +44,11 @@
|
|||||||
1. 下载并安装 [Swift](https://www.swift.org/download/)。
|
1. 下载并安装 [Swift](https://www.swift.org/download/)。
|
||||||
2. 在 VSCode 的插件市场中搜索 `swift` ,安装 [Swift for Visual Studio Code](https://marketplace.visualstudio.com/items?itemName=sswg.swift-lang)。
|
2. 在 VSCode 的插件市场中搜索 `swift` ,安装 [Swift for Visual Studio Code](https://marketplace.visualstudio.com/items?itemName=sswg.swift-lang)。
|
||||||
|
|
||||||
|
## Dart 环境
|
||||||
|
|
||||||
|
1. 下载并安装 [Dart](https://dart.dev/get-dart) 。
|
||||||
|
2. 在 VSCode 的插件市场中搜索 `dart` ,安装 [Dart](https://marketplace.visualstudio.com/items?itemName=Dart-Code.dart-code) 。
|
||||||
|
|
||||||
## Rust 环境
|
## Rust 环境
|
||||||
|
|
||||||
1. 下载并安装 [Rust](https://www.rust-lang.org/tools/install)。
|
1. 下载并安装 [Rust](https://www.rust-lang.org/tools/install)。
|
||||||
|
|||||||
@ -168,13 +168,13 @@ index = hash(key) % capacity
|
|||||||
=== "Dart"
|
=== "Dart"
|
||||||
|
|
||||||
```dart title="simple_hash.dart"
|
```dart title="simple_hash.dart"
|
||||||
[class]{}-[func]{add_hash}
|
[class]{}-[func]{addHash}
|
||||||
|
|
||||||
[class]{}-[func]{mul_hash}
|
[class]{}-[func]{mulHash}
|
||||||
|
|
||||||
[class]{}-[func]{xor_hash}
|
[class]{}-[func]{xorHash}
|
||||||
|
|
||||||
[class]{}-[func]{rot_hash}
|
[class]{}-[func]{rotHash}
|
||||||
```
|
```
|
||||||
|
|
||||||
=== "Rust"
|
=== "Rust"
|
||||||
|
|||||||
@ -129,7 +129,7 @@
|
|||||||
=== "Dart"
|
=== "Dart"
|
||||||
|
|
||||||
```dart title="top_k.dart"
|
```dart title="top_k.dart"
|
||||||
[class]{}-[func]{top_k_heap}
|
[class]{}-[func]{topKHeap}
|
||||||
```
|
```
|
||||||
|
|
||||||
=== "Rust"
|
=== "Rust"
|
||||||
|
|||||||
Reference in New Issue
Block a user