- vừa được xem lúc

[DSA] Binary Heaps, Priority Queue

0 0 3

Người đăng: Tri Lương

Theo Viblo Asia

Binary Heaps

Giới thiệu

Binary heaps giống là complete binary tree (là cây nhị phân được điền đầy đủ ở tất cả các mức từ trái sang phải, ngoài trừ có thể mức cuối cùng), nó có một số tính chất sau:

  • Max Heap: các node cha luôn lớn hơn node con.
  • Min Heap: các node cha nhỏ hơn node con.

image.png

Binary Heap có thể được biểu diễn dưới dạng array như sau:

image.png

Cách xác index của các node cha và con trong 1 array độ như sau:

  • Nếu 1 node nằm index n:
    • Node con bên trái được lưu ở vị trí: 2n + 1
    • Node con bên phải: 2n + 2
  • Vị trí của node cha nếu node con nằm ở index n (làm tròn xuống): (n-1)/2

Khởi tạo

class MaxBinaryHeap { values; constructor() { this.values = []; } insert(element) {} bubbleUp() {} extractMax() {} sinkDown() {}
}

Khởi tạo các method

insert

 insert(element) { this.values.push(element); this.bubbleUp(); } bubbleUp() { let idx = this.values?.length - 1; let element = this.values[idx]; while (idx > 0) { const parentIdx = Math.floor((idx - 1) / 2); const parentElement = this.values[parentIdx]; if (element < parentElement) break; this.values[idx] = parentElement; this.values[parentIdx] = element; idx = parentIdx; } }

extractMax

Trả về giá trị gốc (lớn nhất) → Thay giá trị gốc bằng phần tử cuối cùng của Heap → Sắp xếp lại heap cho đúng.

 extractMax() { const max = this.values[0]; const end = this.values.pop(); this.values[0] = end; this.sinkDown(); return max; } sinkDown() { let idx = 0; let element = this.values[idx]; const length = this.values.length; while (true) { element = this.values[idx]; const leftChildIndx = 2 * idx + 1; const rightChildIndx = 2 * idx + 2; const leftChildElement = this.values[leftChildIndx]; const rightChildElement = this.values[rightChildIndx]; let swap = null; if (leftChildIndx < length) { if (leftChildElement > element) { swap = leftChildIndx; } } if (rightChildIndx < length) { if ( (rightChildElement > element && !swap) || (swap && rightChildElement > leftChildElement) ) { swap = rightChildIndx; } } if (!swap) break; this.values[idx] = this.values[swap]; this.values[swap] = element; idx = swap; } }

Độ phức tạp

Method Big O
Insertion O(log N)
Removal O(log N)
Search O(N)

Priority Queue

Là kiểu dữ liệu giống queue nhưng được sắp xếp dựa vào độ ưu tiên.

Ở đây ta sẽ sử dụng Binary Heap để xây dựng Priority Queue

Khởi tạo

class PriorityQueueNode { value; priority; constructor(value, priority) { this.value = value; this.priority = priority; }
} class PriorityQueue { values: PriorityQueueNode[]; constructor() { this.values = []; } enqueue(value, priority) {} bubbleUp() {} dequeue() {} sinkDown() {}
}

Khởi tạo các method

enqueue

 enqueue(value, priority) { const newNode = new PriorityQueueNode(value, priority); this.values.push(newNode); this.bubbleUp(); } bubbleUp() { let idx = this.values?.length - 1; let element = this.values[idx]; while (idx > 0) { const parentIdx = Math.floor((idx - 1) / 2); const parentElement = this.values[parentIdx]; if (element.priority < parentElement.priority) break; this.values[idx] = parentElement; this.values[parentIdx] = element; idx = parentIdx; } }

dequeue

 dequeue() { const max = this.values[0]; const end = this.values.pop(); this.values[0] = end; this.sinkDown(); return max; } sinkDown() { let idx = 0; let element = this.values[idx]; const length = this.values.length; while (true) { const leftChildIndx = 2 * idx + 1; const rightChildIndx = 2 * idx + 2; element = this.values[idx]; const leftChildElement = this.values[leftChildIndx]; const rightChildElement = this.values[rightChildIndx]; let swap = null; if (leftChildIndx < length) { if (leftChildElement.priority > element.priority) { swap = leftChildIndx; } } if (rightChildIndx < length) { if ( (!swap && rightChildElement.priority > element.priority) || (swap && rightChildElement.priority > leftChildElement.priority) ) { swap = rightChildIndx; } } if (!swap) break; this.values[idx] = this.values[swap]; this.values[swap] = element; idx = swap; } }

Bình luận

Bài viết tương tự

- vừa được xem lúc

Nhập môn lý thuyết cơ sở dữ liệu - Phần 1 : Tổng quan

# Trong bài viết này mình sẽ tập trung vào chủ đề tổng quan về Cơ sở dữ liệu. Phần 1 lý thuyết nên hơi chán các bạn cố gắng đọc nhé, chắc lý thuyết mới làm bài tập được, kiến thức còn nhiều các bạn cứ

0 0 112

- vừa được xem lúc

Cấu trúc dữ liệu và giải thuật: Danh sách liên kết đơn (Singly Linked List)

Danh sách liên kết là 1 cấu trúc dữ liệu được sử dụng để lưu trữ 1 tập hợp các dữ liệu. Danh sách liên kết có các tính chất sau:.

0 0 54

- vừa được xem lúc

Bạn đang lưu dữ liệu dạng cây vào CSDL theo cách nào?

Dữ liệu dạng cây (tree data structure) là dạng dữ liệu rất thường thấy ở các ứng dụng. Những chỗ bạn từng bắt gặp có dùng cấu trúc dạng cây có thể là:.

0 0 82

- vừa được xem lúc

Sự khác nhau giữa biến tham chiếu kiểu List và ArrayList trong Java là gì?

1. Đặt vấn đề.

0 0 47

- vừa được xem lúc

8 kiểu cấu trúc dữ liệu mà mọi lập trình viên cần phải biết.

Trong lĩnh vực Khoa học máy tính, cấu trúc dữ liệu được định nghĩa là những cách để tổ chức và lưu trữ dữ liệu trong máy tính để chúng ta có thể thực hiện các hoạt động trên dữ liệu được lưu trữ đó mộ

0 0 32

- vừa được xem lúc

Data structures: Arrays

Tổng quan:. Tiếp theo trong series Data structures and Algorithms, hôm nay mình sẽ giới thiệu đến các bạn một loại Cấu trúc dữ liệu đơn giản và hay gặp nhất, đó là Array.

0 0 47