關於樹的資料結構(二分搜尋樹,堆和優先佇列)
二分搜尋樹
二分搜尋樹的實現
- 二分搜尋樹不一定是完全二叉樹。
- 左邊都比根節點小,右邊都比根節點大。
- 如果二叉排序樹是平衡的,其查詢效率為O(log2n),近似於折半查詢。如果二叉排序樹完全不平衡,則其深度可達到n,查詢效率為O(n),退化為順序查詢。
宣告樹的結構:
public class BST<E extends Comparable<E>> { //必須具有可比較性
private class Node {
public E e;
public Node left, right;
public Node(E e) {
this.e = e;
left = null;
right = null;
}
}
private Node root;
private int size;
public BST(){
root = null;
size = 0;
}
public int size(){
return size;
}
public boolean isEmpty(){
return size == 0;
}
// 向二分搜尋樹中新增新的元素e
public void add(E e){
root = add(root, e);
}
//方法
...
...
...
}
向以node為根的二分搜尋樹中插入元素e,遞迴演算法。不用遞迴,可採用連結串列型別的方法。
public void add(E e){
root = add(root, e);
}
private Node add(Node node, E e){
if(node == null){
size ++;
return new Node(e);
}
if(e.compareTo(node.e) < 0)
node.left = add(node.left, e);
else if(e.compareTo(node.e) > 0)
node.right = add(node.right, e);
return node;
}
二分搜尋樹的遍歷:
public void preOrder(){
preOrder(root);
}
// 前序遍歷以node為根的二分搜尋樹, 遞迴演算法
private void preOrder(Node node){
if(node == null)
return;
System.out.println(node.e);
preOrder(node.left);
preOrder(node.right);
}
// 二分搜尋樹的非遞迴前序遍歷(棧)
public void preOrderNR(){
if(root == null)
return;
Stack<Node> stack = new Stack<>();
stack.push(root);
while(!stack.isEmpty()){
Node cur = stack.pop();
System.out.println(cur.e);
if(cur.right != null)
stack.push(cur.right);
if(cur.left != null)
stack.push(cur.left);
}
}
// 二分搜尋樹的層序遍歷(佇列)
public void levelOrder(){
if(root == null)
return;
Queue<Node> q = new LinkedList<>();
q.add(root);
while(!q.isEmpty()){
Node cur = q.remove();
System.out.println(cur.e);
if(cur.left != null)
q.add(cur.left);
if(cur.right != null)
q.add(cur.right);
}
}
尋找二分搜尋樹的最小元素及刪除:
// 尋找二分搜尋樹的最小元素
public E minimum(){
if(size == 0)
throw new IllegalArgumentException("BST is empty");
Node minNode = minimum(root);
return minNode.e;
}
// 返回以node為根的二分搜尋樹的最小值所在的節點
private Node minimum(Node node){
if( node.left == null )
return node;
return minimum(node.left);
}
// 從二分搜尋樹中刪除最小值所在節點, 返回最小值
public E removeMin(){
E ret = minimum();
root = removeMin(root);
return ret;
}
// 刪除掉以node為根的二分搜尋樹中的最小節點
// 返回刪除節點後新的二分搜尋樹的根
private Node removeMin(Node node){
if(node.left == null){
Node rightNode = node.right;
node.right = null;
size --;
return rightNode;
}
node.left = removeMin(node.left);
return node;
}
刪除任意一個節點:
// 從二分搜尋樹中刪除元素為e的節點
public void remove(E e){
root = remove(root, e);
}
// 刪除掉以node為根的二分搜尋樹中值為e的節點, 遞迴演算法
// 返回刪除節點後新的二分搜尋樹的根
private Node remove(Node node, E e){
if( node == null )
return null;
if( e.compareTo(node.e) < 0 ){
node.left = remove(node.left , e);
return node;
}
else if(e.compareTo(node.e) > 0 ){
node.right = remove(node.right, e);
return node;
}
else{ // e.compareTo(node.e) == 0
// 待刪除節點左子樹為空的情況
if(node.left == null){
Node rightNode = node.right;
node.right = null;
size --;
return rightNode;
}
// 待刪除節點右子樹為空的情況
if(node.right == null){
Node leftNode = node.left;
node.left = null;
size --;
return leftNode;
}
// 待刪除節點左右子樹均不為空的情況
// 找到比待刪除節點大的最小節點, 即待刪除節點右子樹的最小節點
// 用這個節點頂替待刪除節點的位置
Node successor = minimum(node.right);
successor.right = removeMin(node.right);
successor.left = node.left;
node.left = node.right = null;
return successor;
}
}
leetcode上相關題目
144. 二叉樹的前序遍歷
比較簡單,略
804. 唯一摩爾斯密碼詞
解法:
適合唯一性的資料結構有:
- 二分排序樹:上面解法自己實現了二分樹
- TreeSet:底層是TreeMap
- HashSet:底層是HashMap
- 全域性排序一次,去除掉相鄰重複的
堆和優先佇列
堆的實現
- 堆是完全二叉樹
- 用陣列儲存二叉堆,從索引1開始,則左節點索引=2*父節點的索引,右節點索引=左節點索引+1。
- 從索引0開始,則左節點索引=2*父節點的索引+1,右節點索引=左節點索引+1。
堆的結構:
public class MaxHeap<E extends Comparable<E>> {
private Array<E> data;
public MaxHeap(int capacity){
data = new Array<>(capacity);
}
public MaxHeap(){
data = new Array<>();
}
// 返回堆中的元素個數
public int size(){
return data.getSize();
}
// 返回一個布林值, 表示堆中是否為空
public boolean isEmpty(){
return data.isEmpty();
}
// 返回完全二叉樹的陣列表示中,一個索引所表示的元素的父親節點的索引
private int parent(int index){
if(index == 0)
throw new IllegalArgumentException("index-0 doesn't have parent.");
return (index - 1) / 2;
}
// 返回完全二叉樹的陣列表示中,一個索引所表示的元素的左孩子節點的索引
private int leftChild(int index){
return index * 2 + 1;
}
// 返回完全二叉樹的陣列表示中,一個索引所表示的元素的右孩子節點的索引
private int rightChild(int index){
return index * 2 + 2;
}
}
增加元素,尾部新增,採用siftUp()操作:
// 向堆中新增元素
public void add(E e){
data.addLast(e);
siftUp(data.getSize() - 1);
}
private void siftUp(int k){
while(k > 0 && data.get(parent(k)).compareTo(data.get(k)) < 0 ){
data.swap(k, parent(k));
k = parent(k);
}
}
刪除元素,頂部刪除,尾部元素移到頂部,進行siftDown()操作:
// 取出堆中最大元素
public E extractMax(){
E ret = findMax();
data.swap(0, data.getSize() - 1);
data.removeLast();
siftDown(0);
return ret;
}
private void siftDown(int k){
while(leftChild(k) < data.getSize()){
int j = leftChild(k); // 在此輪迴圈中,data[k]和data[j]交換位置
if( j + 1 < data.getSize() &&
data.get(j + 1).compareTo(data.get(j)) > 0 )
j ++;
// data[j] 是 leftChild 和 rightChild 中的最大值
if(data.get(k).compareTo(data.get(j)) >= 0 )
break;
data.swap(k, j);
k = j;
}
}
heapify操作,使無序陣列變成堆:
public MaxHeap(E[] arr){
data = new Array<>(arr);
for(int i = parent(arr.length - 1) ; i >= 0 ; i --)
siftDown(i);
}
優先佇列:
private interface Queue<E> {
int getSize();
boolean isEmpty();
void enqueue(E e);
E dequeue();
E getFront();
}
private class PriorityQueue<E extends Comparable<E>> implements Queue<E> {
private MaxHeap<E> maxHeap;
public PriorityQueue(){
maxHeap = new MaxHeap<>();
}
@Override
public int getSize(){
return maxHeap.size();
}
@Override
public boolean isEmpty(){
return maxHeap.isEmpty();
}
@Override
public E getFront(){
return maxHeap.findMax();
}
@Override
public void enqueue(E e){
maxHeap.add(e);
}
@Override
public E dequeue(){
return maxHeap.extractMax();
}
}
leetcode上相關題目
堆的典型應用:求1000000萬個元素前100個最大元素。
先排序則需要nlogn,使用100個空間的堆,則時間複雜度為nlogm,m為100。
347. 前 K 個高頻元素
解法:
class Solution {
public int[] topKFrequent(int[] nums, int k) {
TreeMap<Integer, Integer> map = new TreeMap<>();
for(int num: nums){
if(map.containsKey(num))
map.put(num, map.get(num) + 1);
else
map.put(num, 1);
}
PriorityQueue<Integer> pq = new PriorityQueue<>(
(a, b) -> map.get(a) - map.get(b)
);
for(int key: map.keySet()){
if(pq.size() < k)
pq.add(key);
else if(map.get(key) > map.get(pq.peek())){
pq.remove();
pq.add(key);
}
}
int[] num = new int[k];
int i = 0;
while(!pq.isEmpty()){
num[i] = pq.remove();
i++;
}
return num;
}
}
相關文章
- 資料結構-二分搜尋樹資料結構
- 資料結構之PHP二分搜尋樹資料結構PHP
- 【資料結構】搜尋樹資料結構
- 資料結構-佇列-樹資料結構佇列
- 演算法與資料結構之二分搜尋樹演算法資料結構
- python 二叉樹深度優先搜尋和廣度優先搜尋Python二叉樹
- 看得見的資料結構Android版之二分搜尋樹篇資料結構Android
- 資料結構-二叉搜尋樹資料結構
- 【資料結構】二叉搜尋樹!!!資料結構
- 資料結構中的樹(二叉樹、二叉搜尋樹、AVL樹)資料結構二叉樹
- 『演算法與資料結構』優先佇列 二叉堆演算法資料結構佇列
- 資料結構☞二叉搜尋樹BST資料結構
- 資料結構之「二叉搜尋樹」資料結構
- 二分搜尋樹系列之[ 深度優先-層序遍歷 (ergodic) ]Go
- 二分搜尋樹系列之「深度優先-層序遍歷 (ergodic) 」Go
- 堆與優先佇列佇列
- 資料結構-二叉搜尋樹的實現資料結構
- 高階資料結構---堆樹和堆排序資料結構排序
- 二分搜尋樹元素的插入
- Treap——堆和二叉樹的完美結合,價效比極值的搜尋樹二叉樹
- 二叉搜尋樹的結構
- 三、資料結構演算法-棧、佇列、優先佇列、雙端佇列資料結構演算法佇列
- 演算法與資料結構——AVL樹(平衡二叉搜尋樹)演算法資料結構
- 堆、堆排序和優先佇列的那些事排序佇列
- Java關於資料結構的實現:樹Java資料結構
- 資料結構之二叉搜尋樹—Java實現資料結構Java
- 資料結構-二叉樹、堆、圖資料結構二叉樹
- 從二分搜尋到二叉搜尋樹
- 二分搜尋樹(Binary Search Tree)
- 資料結構和演算法-Go實現二叉搜尋樹資料結構演算法Go
- 資料結構-詳解優先佇列的二叉堆(最大堆)原理、實現和應用-C和Python資料結構佇列Python
- 學習資料結構 - 深度優先搜尋 DFS 記錄資料結構
- 資料結構基礎 (程式碼效率優化, 線性表, 棧, 佇列, 陣列,字串,樹和二叉樹,雜湊表)資料結構優化佇列陣列字串二叉樹
- 資料結構——樹狀陣列資料結構陣列
- 【資料結構】【二叉樹】四、二叉搜尋樹的特性(不斷補充)資料結構二叉樹
- 資料結構之PHP(最大堆)實現優先佇列資料結構PHP佇列
- 資料結構高階--二叉搜尋樹(原理+實現)資料結構
- 資料結構:一文看懂二叉搜尋樹 (JavaScript)資料結構JavaScript