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

[DSA] Linked list

0 0 3

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

Theo Viblo Asia

Giới thiệu

Là một kiểu dữ liệu gồm đầu (head), đuôi (tail) và độ dài (length)

Linked list gồm nhiều node, 1 node có 1 giá trị và 1 con trỏ trỏ tới node khác hoặc null

Có 2 loại linked list:

  • Singly linked list (danh sách liên kết đơn)
  • Doubly linked list (danh sách liên kết đôi)

Singly Linked lists (danh sách liên kết đơn)

image.png

Khởi tạo

Tạo 1 class cho node của Singly linked list

class NodeSinglyLinkedList { val: string; next: NodeSinglyLinkedList | null; constructor(val: string) { this.val = val; this.next = null; }
}

Tạo 1 class chứa các method của Singly Linked list

class SinglyLinkedList { head: NodeSinglyLinkedList | null; tail: NodeSinglyLinkedList | null; length: number; constructor() { this.head = null; this.tail = null; this.length = 0; } push(val){} pop(){} unshift(val){} shift(){} get(val){} insert(val, position){} remove(position){} }

Khởi tạo các method

push

 push(val) { const newNode = new NodeSinglyLinkedList(val); if (!this.head) { this.head = newNode; this.tail = this.head; } else { this.tail.next = newNode; this.tail = newNode; } this.length++; return this; }

pop

 pop() { if (!this.head) { return undefined; } let current = this.head; let newTail = current; while (current.next) { newTail = current; current = current.next; } this.tail = newTail; this.tail.next = null; this.length--; return this; }

unshift

 unshift(val) { const newNode = new NodeSinglyLinkedList(val); if (!this.tail) { this.head = newNode; this.tail = this.head; } else { newNode.next = this.head; this.head = newNode; } this.length++; return this; }

shift

 shift() { if (!this.head) { return undefined; } else { const currentHead = this.head; this.head = currentHead.next; } this.length--; return this; }

get

 get(position) { if (position < 0 || position > this.length) return null; let counter = 0; let head = this.head; while (counter < this.length) { if (counter === position) { return head; } head = head.next; counter++; } }

insert

 insert(val, position) { if (position < 0 || position > this.length) return null; const newNode = new NodeSinglyLinkedList(val); if (position === 0) { return this.unshift(val); } if (position === this.length) { return this.push(val); } const prevNode = this.get(position - 1); const prevNodeNext = prevNode.next; prevNode.next = newNode; newNode.next = prevNodeNext; }

remove

 remove(position) { if (position < 0 || position > this?.length) return null; if (position === 0) { return this.shift(); } if (position === this.length - 1) { return this.pop(); } const currentNode = this.get(position); const prevNode = this.get(position - 1); prevNode.next = currentNode.next; return this; }

Doubly Linked lists (danh sách liên kết đôi)

image.png

Khởi tạo

Tạo 1 class cho node của Singly linked list

class NodeDoublyLinkedList { val: string; next: NodeDoublyLinkedList | null; prev: NodeDoublyLinkedList | null; constructor(val: string) { this.val = val; this.next = null; this.prev = null; }
}

Tạo 1 class chứa các method của Singly Linked list

class DoublyLinkedList { head: NodeDoublyLinkedList | null; tail: NodeDoublyLinkedList | null; length: number; constructor() { this.head = null; this.tail = null; this.length = 0; } push(val){} pop(){} unshift(val){} shift(){} get(val){} insert(val, position){} remove(position){} }

Khởi tạo các method

push

 push(val) { const newNode = new NodeDoublyLinkedList(val); if (!this.head) { this.head = newNode; this.tail = newNode; } else { this.tail.next = newNode; newNode.prev = this.tail; this.tail = newNode; } this.length++; return this; }

pop

 pop() { if (!this.head) { return undefined; } const newTail = this.tail.prev; this.tail.next = null; this.tail = newTail; this.length--; return this; }

unshift

 unshift(val) { const newNode = new NodeDoublyLinkedList(val); if (!this.tail) { this.head = newNode; this.tail = this.head; } else { newNode.next = this.head; this.head.prev = newNode; this.head = newNode; } this.length++; return this; }

shift

 shift() { if (!this.head) { return undefined; } else { const currentHead = this.head; this.head = currentHead.next; this.head.prev = null; } this.length--; return this; }

get

 get(position) { if (position < 0 || position > this.length) return null; let counter = 0; let head = this.head; while (counter < this.length) { if (counter === position) { return head; } head = head.next; counter++; } }

insert

 insert(val, position) { if (position < 0 || position > this.length) return null; const newNode = new NodeDoublyLinkedList(val); if (position === 0) { return this.unshift(val); } if (position === this.length) { return this.push(val); } const currentNode = this.get(position); const prevNode = this.get(position - 1); prevNode.next = newNode; newNode.prev = prevNode; newNode.next = currentNode; currentNode.prev = newNode; this.length++; return this; }

remove

 remove(position) { if (position < 0 || position > this?.length) return null; if (position === 0) { return this.shift(); } if (position === this.length - 1) { return this.pop(); } const prevNode = this.get(position - 1); const nextNode = this.get(position + 1); prevNode.next = nextNode; nextNode.prev = prevNode; this.length--; return this; }

Kết luận

Từ việc triền khai các method ở trên ta có được bảng so sánh độ phức tạp của 2 loại linked list như sau:

Method Singly Doubly
push O(1) O(1)
pop O(n) O(1)
unshift O(1) O(1)
shift O(1) O(1)
get O(n) O(n)
insert O(n) O(n)
remove O(n) O(n)

Bình luận

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

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

Đôi chút về cấu trúc dữ liệu và thuật toán

Theo anh Phạm Huy Hoàng (toidicodedao) đã viết trong blog của mình là kiến thức trong ngành IT có 2 loại, một là càng để lâu thì càng cũ, lạc hậu và trở lên vô dụng. Hai là càng để lâu thì càng có giá trị thậm chí ngày càng có giá.

0 0 43

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

Cấu trúc dữ liệu Merkle Tree

Cây Merkle là một cây nhị phân có thứ tự được xây dựng từ một dãy các đối tượng dữ liệu (d1, d2,...,dn) sử dụng hàm băm h. Các “lá” của cây là các giá trị băm h(di) đối với 1 ≤ i ≤ n.

0 0 66

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

Cách xây dựng cấu trúc dữ liệu Stack và Queue.

Mở đầu. Hello các bạn, hôm nay mình sẽ chia sẻ với các bạn cách để có thể tự xây dựng 2 loại cấu trúc dữ liệu stack(ngăn xếp) và queue(hàng đợi) sử dụng mảng trong C++;.

0 0 43

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

Tối ưu ứng dụng với cấu trúc dữ liệu cơ bản

Ở Việt Nam có một nghịch lý ai cũng biết: hầu hết sinh viên ngành CNTT đều đã học cấu trúc dữ liệu và giải thuật, thuộc các môn bắt buộc. Thế nhưng lại rất hiếm khi ứng dụng vào công việc hoặc bị loại

0 0 49

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

Chương 1: Introduction - Analysis of Algorithrms

Trong bài viết này mình sẽ nói về cách chúng ta sẽ sử dụng để phân tích và so sánh các loại thuật toán khác nhau. 1.

0 0 27

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

Chương 1: Introduction - Các khái niệm cơ bản

Lời nói đầu. Trước khi có máy tính, đã có các thuật toán.

0 0 25