数据结构
栈
-
受限的线性结构
-
后进先出 LIFO
-
栈定、栈底
基于数组实现
属性
item: []
常见方法
function Stack() {
// 属性
this.item = [];
// push
Stack.prototype.push = function (element) {
this.item.push(element);
};
// pop
Stack.prototype.pop = function () {
return this.item.pop();
};
// peek
Stack.prototype.peek = function () {
return this.item[this.item.length - 1];
};
// isEmpty
Stack.prototype.isEmpty = function () {
return this.item.length === 0;
};
// size
Stack.prototype.size = function () {
return this.item.length;
};
// toString
Stack.prototype.toString = function () {
let res = "";
for (let i = 0; i < this.item.length; i++) {
res += this.item[i] + " ";
}
return res;
};
}
应用
十进制转二进制
function dec2bin(decNumber) {
// 栈对象-保存余数
const stack = new Stack();
while (decNumber > 0) {
stack.push(decNumber % 2);
// 获取商,作为下一次的decNumber
decNumber = Math.floor(decNumber / 2);
}
// 从栈中取出0和1
let binRes = "";
while (!stack.isEmpty()) {
binRes += stack.pop();
}
return binRes;
}
链表
数组的缺点
- 连续的内存空间,大部分编程语言的数组大小固定,当不能满足容量需求时,需要扩容,而扩容是非常消耗性能
- 在头部或中间插入数据,需要进行大量元素的位移
链表的出现
优点
相对于数组,链表有一些优点:
- 内存空间不必连续~,灵活
- 不用创建时确定大小
- 插入和删除,时间复杂度可以达到 O(1)
缺点
而对于数组,链表也有一些缺点:
- 访问任何一个元素,都需要从头开始
总结
侧重查询 | 选数组 |
---|---|
侧重插入和删除 | 选链表 |
链表和火车的结构非常相似~
实现
单向链表
属性
- head
- length
- node 的 data
- node 的 next
function LinkedList() {
// 属性
this.head = null;
this.length = 0;
// 内部类 节点
function Node(data) {
this.data = data;
this.next = null;
}
}
常见操作
- append
- insert
- get
- update
- remove
- removeAt
- indexOf
- size
- empty
- toString
append
- 创建新节点
- 添加的是第一个节点?
- 找出最后节点
- 长度+1
LinkedList.prototype.append = function (data) {
// 创建新节点
const newNode = new Node(data);
// 添加的是第一个节点?
if (this.length === 0) {
this.head = newNode;
} else {
// 不是
// 找出最后节点
let last = this.head;
while (last.next) {
last = last.next;
}
// 最后节点指向新加节点
last.next = newNode;
}
// 长度+1
this.length += 1;
};
toString
LinkedList.prototype.toString = function (params) {
let current = this.head;
let res = "";
// 遍历节点
while (current) {
res += current.data + " ";
current = current.next;
}
return res;
};
测试
const list = new LinkedList()
list.append('abc')
list.append('abb')
list.append('ccc')
console.log(list)
insert
- position 越界判断
- 根据 data 创建新节点
- case
- 长度+1
LinkedList.prototype.insert = function (position, data) {
// position越界判断
if (position < 0 || position > this.length) {
return false;
}
// 根据data创建新节点
const newNode = new Node(data);
// case1 插入position = 0 的位置
if (position === 0) {
newNode.next = this.head;
this.head = newNode;
} else {
// case2 插入position > 1 的位置
// 找到位置
let index = 0;
// 记录目标节点
let current = this.head;
// 记录目标节点的上一个节点
let previous = null;
while (index++ < position) {
previous = current;
current = current.next;
}
newNode.next = current;
previous.next = newNode;
}
// 长度+1
this.length += 1;
return true;
};
其中,插入到最后的情况,case2 已经包含,而且,要是想插入到最后,用append呗
测试
// 测试
const list = new LinkedList();
// append
list.append("abc");
list.append("abb");
list.append("ccc");
// insert
list.insert(0, "zsf");
list.insert(3, "hhh");
list.insert(100, "lll");
console.log(list);
get
- position 越界判断,=也不行
- 获取 data
LinkedList.prototype.get = function (position) {
// position越界判断,=也不行
if (position < 0 || position >= this.length) {
return null;
}
// 获取data
let current = this.head;
let index = 0;
while (index++ < position) {
current = current.next;
}
return current.data;
};
测试
const list = new LinkedList();
// append
list.append("abc");
list.append("abb");
list.append("ccc");
// insert
list.insert(0, "zsf");
list.insert(3, "hhh");
list.insert(100, "lll");
// get
console.log(list.get(0));
console.log(list.get(2));
console.log(list.get(9));
console.log(list);
indexOf
LinkedList.prototype.indexOf = function (data) {
let current = this.head;
let index = 0;
while (current) {
if (current.data === data) {
return index;
}
current = current.next;
index += 1;
}
return -1;
};
测试
const list = new LinkedList();
console.log(list.indexOf("zsf"));
console.log(list.indexOf("hhh"));
console.log(list.indexOf("ll"));
update
- position 越界判断,=也不行
- 找到目标节点
LinkedList.prototype.update = function (position, data) {
// position越界判断,=也不行
if (position < 0 || position >= this.length) {
return false;
}
// 找到目标节点
let current = this.head;
let index = 0;
while (index++ < position) {
current = current.next;
}
current.data = data;
return true;
};
测试
const list = new LinkedList();
console.log(list.update(0, "hhh"));
removeAt
position 越界判断,=也不行
- 第一个节点?
- 目标节点的上一个节点
- 长度-1
LinkedList.prototype.removeAt = function (position) {
// position越界判断,=也不行
if (position < 0 || position >= this.length) {
return null;
}
// 第一个节点?
// 目标节点
let current = this.head;
if (position === 0) {
this.head = this.head.next;
} else {
// 目标节点的上一个节点
let previous = null;
let index = 0;
while (index++ < position) {
previous = current;
current = current.next;
}
previous.next = current.next;
}
// 长度-1
this.length -= 1;
return current.data;
};
测试
const list = new LinkedList();
console.log(list.removeAt(0));
remove
- 根据 data 获取位置
- 根据位置删除
LinkedList.prototype.remove = function (data) {
// 根据data获取位置
const position = this.indexOf(data);
// 根据位置删除
return this.removeAt(position);
};
测试
const list = new LinkedList();
console.log(list.remove("abc"));
isEmpty
LinkedList.prototype.isEmpty = function () {
return this.length === 0;
};
size
LinkedList.prototype.size = function () {
return this.length;
};
反转单链表
/**
* Definition for singly-linked list.
* function ListNode(val) {
* this.val = val;
* this.next = null;
* }
*/
/**
* @param {ListNode} head
* @return {ListNode}
*/
var reverseList = function (head) {
let temp = null;
let pre = null;
let cur = head;
while (cur) {
cur = head.next;
head.next = pre;
prev = head;
head = p;
}
return pre;
};
示例
输入: 1->2->3->4->5->NULL
输出: 5->4->3->2->1->NULL
复盘:
时间复杂度 O(N):遍历链表使用线性大小时间。
空间复杂度 O(1):变量 pre 、temp、cur 使用常数大小额外空间。
1.pre 指向 null,cur 指向头节点
pre = null;
cur = head;
2.暂存后继节点,修改引用指向
temp = cur.next;
cur.next = pre;
3.暂存当前节点,访问下一节点
pre = cur;
cur = temp;
4.循环第 2~3 步
temp = cur.next;
cur.next = pre;
pre = cur;
cur = temp;
5.循环到最后两步,cur、temp 回收,返回 pre 是新的头节点
双向链表
单向链表有个明显的不足:
回到前一个节点
相对于单向链表,双向也有一些缺点:
- 插入/删除某个节点,要处理4 个引用
- 一个节点内存空间更大
属性
- head
- tail
- length
- node 的 data
- node 的 prev
- node 的 next
function DoublyLinkedList() {
// 属性
this.head = null;
this.tail = null;
this.length = 0;
// 内部类 节点
function Node(data) {
this.data = data;
this.prev = null;
this.next = null;
}
}
常见操作
- append
- insert
- backwrd/forward
append
DoublyLinkedList.prototype.append = function (data) {
// 创建新节点
const newNode = new Node(data);
// 添加的是第一个节点?
if (this.length === 0) {
this.head = newNode;
this.tail = newNode;
} else {
// 不是
newNode.prev = this.tail;
this.tail.next = newNode;
this.tail = newNode;
}
// 长度+1
this.length += 1;
};
insert
DoublyLinkedList.prototype.insert = function (position, data) {
// position越界判断
if (position < 0 || position > this.length) {
return false;
}
// 根据data创建新节点
const newNode = new Node(data);
// case1 原来链表是否为空
if (this.length === 0) {
this.head = newNode;
this.tail = newNode;
} else {
// case2 插入position = 0 的位置
if (position === 0) {
this.head.prev = newNode;
newNode.next = this.head;
this.head = newNode;
} else if (position === this.length) {
// case3 插入position = this.length 的位置
newNode.prev = this.tail;
this.tail.next = newNode;
this.tail = newNode;
} else {
let current = this.head;
let index = 0;
// 找到目标节点
while (index++ < position) {
current = current.next;
}
newNode.next = current;
newNode.prev = current.prev;
current.prev.next = newNode;
current.prev = newNode;
}
}
// 长度+1
this.length += 1;
return true;
};
backward\forward
// backward向后遍历
DoublyLinkedList.prototype.backward = function (params) {
let current = this.head;
while (current) {
current = current.next;
}
return current;
};
// forward
DoublyLinkedList.prototype.forward = function (params) {
let current = this.tail;
while (current) {
current = current.prev;
}
return current;
};
其它方法类似,就是比单向链表修改多个指向
队列
- 先进先出 FIFO
- 前删后增
应用场景
线程队列
事件队列
实现
- 基于数组
- 基于链表
常见操作
- enQueue
- deQueue
- front
function Queue() {
// 属性
this.items = [];
// 方法
// enQueue
Queue.prototype.enQueue = function (element) {
this.item.push(element);
};
// deQueue
Queue.prototype.deQueue = function () {
return this.items.shift();
};
// front
Queue.prototype.front = function () {
return this.items[0];
};
}
击鼓传花
几个人围成一圈,数到某个数字的人自动淘汰
最后剩下的这个人会获得胜利,问,胜利者原来在哪一个位置
function passGame(nameList, num) {
// 创建队列
let queue = new Queue();
// 将所有人加入队列
for (let i = 0; i < nameList.length; i++) {
queue.enQueue(nameList[i]);
}
// 数数字
// 当队列长度不为1时一直循环
while (queue.size() > 1) {
// num数字之前的人重新放入队列的末尾
for (let i = 0; i < num - 1; i++) {
queue.enQueue(queue.deQueue);
}
// num对应的的人,从队列中删除~
queue.deQueue();
}
return nameList.indexof(queue.front());
}
优先级队列
插入一个元素时会考虑该数的优先级
应用
- 登机
- 急诊科
- 线程
实现
function PriorityQueue() {
// 属性
this.items = [];
function QueueElement(element, priority) {
this.element = element;
this.priority = priority;
}
// 方法
// enQueue
PriorityQueue.prototype.enQueue = function (element, priority) {
// 创建QueueElement对象
const queueElement = new QueueElement(element, priority);
// 判断队列是否为空
if (this.isEmpty()) {
this.items.push(queueElement);
} else {
// 记录是否有机会添加
let added = false;
for (let i = 0; i < this.items.length; i++) {
if (queueElement.priority < this.items[i].priority) {
this.items.splice(i, 0, queueElement);
added = true;
break;
}
}
// 如果优先级不如队列中其它元素,直接放最后
if (!added) {
this.items.push(queueElement);
}
}
};
// deQueue
PriorityQueue.prototype.deQueue = function () {
return this.items.shift();
};
// front
PriorityQueue.prototype.front = function () {
return this.items[0];
};
}
树
完美二叉树
除了最后一层的叶子节点外,每层节点都有 2 个子节点,又叫满二叉树
完全二叉树
- 除最后一层外,其它各层的节点数都达到最大个数
- 最后一层从左向右的叶节点连续存在,只缺右侧若干节点
- 完美二叉树是特殊的完全二叉树
二叉搜索树
Binary Search Tree(BST)
特点
- 非空左子树所有的键值小于其根节点的键值
- 非空右子树所有的键值大于其根节点的键值
- 左右子树本身也是二叉搜索树
举个例子
封装
function BinarySeacherTree() {
function Node(key) {
this.key = key;
this.left = null;
this.right = null;
}
// 属性-根结点
this.root = null;
}
常见操作
- insert
- search(优势)
- remove(复杂)
- inOrderTraverse
- preOrderTraverse
- POSTOrderTraverse
- min
- max
insert
// insert
BinarySeacherTree.prototype.insert = function (key) {
// 根据key创建节点
const newNode = Node(key);
// case1 根结点是否有值
if ((this.root = null)) {
this.root = newNode;
} else {
this.insertNode(this.root, newNode);
}
};
// insertNode 用于递归插入
BinarySeacherTree.prototype.insertNode = function (node, newNode) {
if (newNode.key < node.key) {
// 向左查找
if (node.left == null) {
// 为空?
node.left = newNode;
} else {
this.insertNode(node.left, newNode);
}
} else {
// 向右查找
if (node.right == null) {
// 为空?
node.right = newNode;
} else {
this.insertNode(node.right, newNode);
}
}
};
遍历
- 先序遍历(先处理,再找子)
- 中序遍历
先序遍历
// preOrderTraversal
BinarySeacherTree.prototype.preOrderTraversal = function (handler) {
this.preOrderTraversalNode(this.root, handler);
};
// preOrderTraversalNode 用于节点递归遍历
BinarySeacherTree.prototype.preOrderTraversalNode = function (node, handler) {
if (node != null) {
// 1.处理经过的节点
handler(node.key);
// 2.处理经过节点的左子节点
this.preOrderTraversalNode(node.left, handler);
// 3.处理经过节点的右子节点
this.preOrderTraversalNode(node.right, handler);
}
};
中序遍历
// 中序遍历 midOrderTraversal
BinarySeacherTree.prototype.midOrderTraversal = function (handler) {
this.midOrderTraversalNode(this.root, handler);
};
// preOrderTraversalNode 用于中 序节点递归遍历
BinarySeacherTree.prototype.midOrderTraversalNode = function (node, handler) {
if (node != null) {
// 1.先处理左子树中的节点
this.midOrderTraversalNode(node.left, handler);
// 2.处理节点
handler(node.key);
// 3.处理右子树中的节点
this.midOrderTraversalNode(node.right, handler);
}
};
后序遍历
// 后序遍历
BinarySeacherTree.prototype.postOrderTraversal = function (handler) {
this.postOrderTraversalNode(this.root, handler);
};
// preOrderTraversalNode 用于中序节点递归遍历
BinarySeacherTree.prototype.postOrderTraversalNode = function (node, handler) {
if (node != null) {
// 1.先处理左子树中的节点
this.postOrderTraversalNode(node.left, handler);
// 2.处理右子树中的节点
this.postOrderTraversalNode(node.right, handler);
// 3.处理节点
handler(node.key);
}
};
搜索
- 最大/最小值
- 特定的值
最值
// 寻找最值
BinarySeacherTree.prototype.max = function () {
// 获取根结点
let node = this.root;
// 直到节点没右子节点
while (node.right != null) {
node = node.right;
}
return node.key;
};
BinarySeacherTree.prototype.min = function () {
// 获取根结点
let node = this.root;
// 直到节点没左子节点
while (node.left != null) {
node = node.left;
}
return node.key;
};
特定值
// 搜索某个key
BinarySeacherTree.prototype.search = function (key) {
let node = this.root;
// 循环搜索
while (node != null) {
if (key < node.key) {
node = node.left;
} else if (key > node.key) {
node = node.right;
} else {
return true;
}
}
return false;
};
删除
比较复杂~
-
找到要删除的节点
-
分 3 情况:
-
没有子节点
-
有一个子节点
左节点?右节点?
左子节点?右子节点?
-
有两个子节点
-
找到要删除的节点
// 1.找到需要删除的节点
let current = this.root;
// 它的父节点
let parent = null;
// 判断是否删除的是左子节点
let isLeftChild = false;
while (current.key != key) {
parent = current;
if (key < current.key) {
isLeftChild = true;
} else {
isLeftChild = false;
current = current.right;
}
// 到最后都没找到~
if (current == null) {
return false;
}
}
没有子节点
if (current.left == null && current.right) {
if (current == this.root) {
this.root = null;
} else if (isLeftChild) {
parent.left = null;
} else {
parent.right = null;
}
}
有一个子节点
别忘记考虑删除的是根的情况~
// 删除的是有一个子节点的父节点
else if (current.right == null) {
if (current == this.root) {
this.root = current.left
} else if (isLeftChild) {
parent.left = current.left
} else {
parent.right = current.left
}
} else if (current.left == null) {
if (current == this.root) {
this.root = current.right
} else if (isLeftChild) {
parent.left = current.right
} else {
parent.right = current.right
}
}
有两个子节点
删除后保证还是二叉搜索树
删除节点 9
- 替代节点 9,节点8 符合;
- 替代节点 9,节点10 符合;
删除节点 7
- 替代节点 7,节点5 符合;
- 替代节点 7,节点8 符合;
删除节点 15
- 替代节点 15,节点14 符合;
- 替代节点 15,节点18 符合;
规律总结:
若用 current 表示需要删除的节点,则合适的替代节点是:
- current 左子树中比 current小一点点的节点,即 current左子树中的最大值;
- current 右子树中比 current大一点点的节点,即 current右子树中的最小值;
在二叉搜索树中,这两个特殊的节点有特 殊的名字:
- 前驱。比如下图中的节点 5 就是节点 7 的前驱;
- 后继。比如下图中的节点 8 就是节点 7 的后继;
如何查找前驱和后继?
后继:在 current 的右子树中查找最小值,即在 current 的右子树中一直向左遍历查找;
前驱:在 current 的左子树中查找最大值,即在 current 的左子树中一直向右遍历查找
后继代码的封装
//封装查找后继的方法
BinarySearchTree.prototype.getSuccessor = function (delNode) {
//1.定义变量,保存找到的后继
let successor = delNode;
let current = delNode.right;
let successorParent = delNode;
//2.循环查找current的右子树节点
while (current != null) {
successorParent = successor;
successor = current;
current = current.left;
}
//3.判断寻找到的后继节点是否直接就是删除节点的right节点
if (successor != delNode.right) {
successorParent.left = successor.right;
successor.right = delNode.right;
}
return successor;
};
代码
else{
//1.获取后继节点
let successor = this.getSuccessor(current)
//2.判断是否根节点
if (current == this.root) {
this.root = successor
}else if (isLeftChild){
parent.left = successor
}else{
parent.right = successor
}
//3.将后继的左子节点改为被删除节点的左子节点
successor.left = current.left
}
算法
排序
选择排序
找最小,放开头;
剩下找最小,放剩下开头,重复这步;
案例 [2, 3, 5, 4, 1]
第一轮 13542
第二轮 12543
第三轮 12453 12354
第四轮 12345
/*
* @Description: 选择排序
* @Author: SiFeng Zhai
* @Date: 2022-09-26 19:04:25
* @LastEditors: SiFeng Zhai
* @LastEditTime: 2022-09-26 19:19:24
*/
function selectionSort(arr) {
// 边界
if (arr === undefined || arr.length < 2) {
return arr;
}
for (let i = 0; i < arr.length - 1; i++) {
let minIndex = i;
for (let j = i + 1; j < arr.length; j++) {
// 一轮比较,更新最小下标
minIndex = arr[j] < arr[minIndex] ? j : minIndex;
}
swap(arr, i, minIndex);
}
return arr;
}
function swap(arr, i, j) {
let temp = arr[i];
arr[i] = arr[j];
arr[j] = temp;
}
// 测试
const arr1 = [2, 3, 5, 4, 1];
console.log(selectionSort(arr1));
冒泡排序
找最大,放尾部
剩下找最大,放剩下尾部,重复