|
1 | 1 | # Linked List
|
2 | 2 |
|
3 |
| -In computer science, a linked list is a linear collection |
4 |
| -of data elements, in which linear order is not given by |
5 |
| -their physical placement in memory. Instead, each |
6 |
| -element points to the next. It is a data structure |
7 |
| -consisting of a group of nodes which together represent |
8 |
| -a sequence. Under the simplest form, each node is |
9 |
| -composed of data and a reference (in other words, |
10 |
| -a link) to the next node in the sequence. This structure |
11 |
| -allows for efficient insertion or removal of elements |
12 |
| -from any position in the sequence during iteration. |
13 |
| -More complex variants add additional links, allowing |
14 |
| -efficient insertion or removal from arbitrary element |
15 |
| -references. A drawback of linked lists is that access |
16 |
| -time is linear (and difficult to pipeline). Faster |
17 |
| -access, such as random access, is not feasible. Arrays |
18 |
| -have better cache locality as compared to linked lists. |
19 |
| - |
20 |
| - |
21 |
| - |
22 |
| -## References |
23 |
| - |
24 |
| -- [Wikipedia](https://en.wikipedia.org/wiki/Linked_list) |
25 |
| -- [YouTube](https://www.youtube.com/watch?v=njTh_OwMljA&index=2&t=1s&list=PLLXdhg_r2hKA7DPDsunoDZ-Z769jWn4R8) |
| 3 | +## Description |
| 4 | + |
| 5 | +A linked list is a linear colletion of elements _in sequential order_. Imagine a chain, with a start and a end, with the chain links connected to each other. A linked list can be visualized as below: |
| 6 | + |
| 7 | + |
| 8 | + |
| 9 | +A linked list is composed of a smaller data structure usually called a `Node`. The node object contains the `value` of the element itself, and also a pointer to the `next` node in the chain. By utilizing `node.next`, you can traverse the list. The beginning of a linked list is denoted with `head` and the last element is `tail`. |
| 10 | + |
| 11 | +In a singly-linked list, the link is established in _one direction_ only, where the node only keeps track of the next node in the chain. This means that with a singly-linked list, you can only traverse in the direction of the tail. On the other hand, a node in a doubly-linked list keeps track of both `next` and `previous` which allows you to traverse in both directions (towards `head` and `tail`). |
| 12 | + |
| 13 | +## Implementation |
| 14 | + |
| 15 | +In this exercise, implement the following functions for the `LinkedListNode` and `LinkedList` classes: |
| 16 | + |
| 17 | +- `LinkedListNode` |
| 18 | + - `constructor()` |
| 19 | + - Write a method that instantiates the node. |
| 20 | + - The node takes `value` and `next`. |
| 21 | +- `LinkedList` |
| 22 | + - `constructor()` |
| 23 | + - Write a method that instantiates the list. |
| 24 | + - The instance variables `head` and `tail` should be set to `null`. |
| 25 | + - `prepend(value)` |
| 26 | + - Write a method that inserts the `value` at the beginning of the linked list. |
| 27 | + - `append(value)` |
| 28 | + - Write a method that inserts the `value` at the end of the linked list. |
| 29 | + - `delete(value)` |
| 30 | + - Write a method that deletes the `value` in the linked list. |
| 31 | + - `find({ value })` |
| 32 | + - Write a method that returns the `node` that contains the `value`. |
| 33 | + - `insertAfter(value, targetValue)` |
| 34 | + - Write a method that inserts the `node` after the `node` that contains the `targetValue`. |
| 35 | + - `deleteHead()` |
| 36 | + - Write a method that deletes the first element in the linked list. |
| 37 | + - `deleteTail()` |
| 38 | + - Write a method that deletes the last element in the linked list. |
| 39 | + |
| 40 | +The most important operations are `prepend/append` for adding data, `delete` for removing data, and `find` for retrieving data. |
| 41 | + |
| 42 | +## Detailed Walkthrough |
| 43 | + |
| 44 | +Let's start by opening up a terminal window and running `npm run watch`. This will start our TDD framework that will automatically re-run the tests when you save any changes to your file. |
| 45 | + |
| 46 | +* In your code editor, navigate to `/algorithms/src/data-structures/LinkedListNode.js`. |
| 47 | +* In the constructor method, write `this.value = value` and save the changes. |
| 48 | +* You will see the tests run automatically in your terminal: |
| 49 | + |
| 50 | + |
| 51 | + |
| 52 | +> Jest Tip: If you need to filter by the exercise name, press `p`. |
| 53 | +> (press `w` for the list of available commands). |
| 54 | +
|
| 55 | +The output tells you that three tests are failing: |
| 56 | +* ✕ should create list node with value (10ms) |
| 57 | +* ✕ should create list node with object as a value (2ms) |
| 58 | +* ✕ should link nodes together |
| 59 | +* ✓ should convert node to string |
| 60 | +* ✓ should convert node to string with custom stringifier (1ms) |
| 61 | + |
| 62 | +The code for the last two tests (`toString()` tests) have been provided for you in the skeleton. Your task is to make the three failing tests to pass. |
| 63 | + |
| 64 | +### `LinkedListNode` |
| 65 | +* The test description says `› should create list node with value` |
| 66 | +* The output tells us where the test is failing: `expect(node.next).toBeNull();`. It also tells us `expect(received).toBeNull()` and `Received: undefined`. |
| 67 | +* In this case, the test is `expect`ing `node.next` to be `null`. However, the test received `undefined`. |
| 68 | +* Since `next` is undefined, let's update our `constructor()` in `LinkedListNode.js`: |
| 69 | + ``` |
| 70 | + constructor(value, next = null) { |
| 71 | + this.value = value; |
| 72 | + this.next = next; |
| 73 | + } |
| 74 | + ``` |
| 75 | +* When you save the file, you should see all your tests for `LinkedListNode` pass! |
| 76 | +* Often, many tests can depend on the same part of the code. In this case, solving for one test case solved all others. |
| 77 | + |
| 78 | +Now, let's open up `LinkedList.js` in our editor. |
| 79 | + |
| 80 | +### `constructor()` |
| 81 | +* A linked list should always have a `head` and a `tail`. |
| 82 | +* The test tells us: `expect(linkedList.head).toBeNull();`. Let's create the instance variables: |
| 83 | + ``` |
| 84 | + constructor() { |
| 85 | + this.head = null; |
| 86 | + this.tail = null; |
| 87 | + } |
| 88 | + ``` |
| 89 | + |
| 90 | +### `prepend()` |
| 91 | +* The `prepend` method inserts the item at the beginning of the list. |
| 92 | +* Prepending an item is a simple operation: |
| 93 | + * Create a new `Node`. |
| 94 | + * Set the new `node`'s `next to be the current head node. |
| 95 | + * Update the `head` to point at the new node. |
| 96 | +* We also have to consider a case where this is the first item stored in the list. In that case, we also set the tail to the same node. |
| 97 | + |
| 98 | +### `append()` |
| 99 | +* The `append` method inserts the item at the end of the list. |
| 100 | +* Operations: |
| 101 | + * Create a new `Node`. |
| 102 | + * Set the tail's `next` to be the new node. |
| 103 | + * Update `tail` to point at the new node. |
| 104 | +* Again, take into consideration when this is the first item stored in the list. |
| 105 | + |
| 106 | +### `find({ value })` |
| 107 | +* The `find` method returns the node with the target value. |
| 108 | +* A node can be visited by utilizing a loop and `node.next`. |
| 109 | +* The following snippet traverses the entire list: |
| 110 | + ``` |
| 111 | + let currentNode = node; |
| 112 | +
|
| 113 | + while (currentNode.next) { |
| 114 | + currentNode = node.next |
| 115 | + console.log(currentNode.value) |
| 116 | + } |
| 117 | + ``` |
| 118 | +* Think about how the above concept can be applied to find the target node. |
| 119 | + |
| 120 | +### `insertAfter(value, insertValue)` |
| 121 | +* The `insertAfter` method stores the `insertValue` right after the node with the `value`. |
| 122 | +* Operation: |
| 123 | +  |
| 124 | + * Create a new `Node`. |
| 125 | + * Find the node with the target value. |
| 126 | + * Set the found node's `next` to the new node. |
| 127 | + * Set new node's `next` to the found node's `next. |
| 128 | +* Utilize the previously written `find` method to find the node. |
| 129 | + |
| 130 | +### `delete(value)` |
| 131 | +* The `delete` method removes the first node with the specified value. |
| 132 | +* Operations: |
| 133 | +  |
| 134 | + * Traverse the list and locate the node before the target node. |
| 135 | + * Remove the target node that is placed after the found node. |
| 136 | + * In a linked list, you don't have to delete the actual node. |
| 137 | + * Deleting means removing the reference to that node from the list. |
| 138 | + * The above diagram shows that all we have to do is set the target node's previous node's `next` to point at the target node's `next`. |
| 139 | + * This means that we must locate the node right before the target node. Think about how you would be able to tell when the next node will be the target node. |
| 140 | + * Also, take into consideration if the `head` or `tail` node is the node to be deleted. |
| 141 | +* This is the most complex operation in a linked list. Drawing the operation on paper can help you visualize the problem. Don't hesitate to reach out to #basic-algorithms channel on Slack if you are stuck. |
| 142 | + |
| 143 | +### `deleteHead()` / `deleteTail()` |
| 144 | +* The `deleteHead/Tail` methods are utility methods for common operations. |
| 145 | + |
| 146 | +## Time and Space Complexity Analysis |
| 147 | + |
| 148 | +This section will be released in the future when the Big O notation tutorials are written. Please join #basic-algorithms channel for the update notifications. |
0 commit comments