Skip to content

Commit 124c9f4

Browse files
author
Kevin K. Lee
authored
Implement Linked List exercise (#2)
* Update LinkedList skeleton, solution and tests * Remove the Comparator helper * Publish the initial LinkedList walkthrough draft * Utilize README in /solutions folder * Make correction for node removal in linked list * Add possible curriculum schedules * Add the linked list walkthrough to /skeletons and /src folders
1 parent 3ad2c73 commit 124c9f4

18 files changed

+826
-188
lines changed

README.md

+58-1
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,64 @@ The problem sets below contain information on key data structures, sorting/searc
4444

4545
### Schedule
4646

47-
The plan is to meet in #algorithms-curriculum channel on [Operation Code Slack](https://operationcode.org), and solve one or two exercises a week. The curriculum will reset and restart after solving every exercise. The exercises will be in increasing difficulty, and though you can jump in and out, be aware that some are built on top of previously built data structures.
47+
These schedules are guidelines. Please approach them at your own pace.
48+
49+
#### Regular
50+
Recommended track for beginners who can devote 2-4 hours a day.
51+
52+
* Day 1
53+
- Queue (1 Hour)
54+
- Stack (1 Hour)
55+
* Day 2
56+
- Linked List (4 Hours)
57+
* Day 3
58+
- Hash Table (4 Hours)
59+
* Day 4
60+
- Selection Sort (30 Minutes)
61+
- Insertion Sort (30 Minutes)
62+
- Bubble Sort (2 Hours)
63+
* Day 5
64+
- Merge Sort (4 Hours)
65+
* Day 6
66+
- Quicksort (4 Hours)
67+
* Day 6
68+
- Binary Search (2 Hours)
69+
* Day 7
70+
- Depth-First Search (2 Hour)
71+
* Day 8
72+
- Breadth-First Search (2 Hour)
73+
* Day 9
74+
- Binary Search Tree (4 Hours)
75+
* Day 10
76+
- Binary Search Tree (4 Hours)
77+
* Day 11
78+
- Fibonacci Number (1 Hour)
79+
- Prime Number (1 Hour)
80+
- Factorial (2 Hours)
81+
82+
#### Quick
83+
Recommended for someone who is comfortable coding and can devote ~6 hours a day.
84+
85+
* Day 1
86+
- Queue (30 Minutes)
87+
- Stack (30 Minutes)
88+
- Linked List (2 Hours)
89+
- Hash Table (2 Hours)
90+
* Day 2
91+
- Selection Sort (30 Minutes)
92+
- Insertion Sort (30 Minutes)
93+
- Bubble Sort (1 Hour)
94+
- Merge Sort (4 Hours)
95+
- Quicksort (2 Hours)
96+
* Day 3
97+
- Binary Search (1 Hour)
98+
- Depth-First Search (1 Hour)
99+
- Breadth-First Search (1 Hour)
100+
- Factorial (30 Minutes)
101+
- Fibonacci Number (1 Hour)
102+
- Prime Number (1 Hour)
103+
* Day 4
104+
- Binary Search Tree (6 Hours)
48105

49106
### Folders
50107

assets/linked-list-delete.png

42.2 KB
Loading

assets/linked-list-insert-after.png

56.8 KB
Loading

assets/linked-list-overview.png

30.4 KB
Loading

assets/linked-list-tdd-console.png

235 KB
Loading

assets/linked-list-tdd.png

235 KB
Loading

assets/linkedlist.png

13.4 KB
Loading
+146-23
Original file line numberDiff line numberDiff line change
@@ -1,25 +1,148 @@
11
# Linked List
22

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-
![Linked List](https://upload.wikimedia.org/wikipedia/commons/6/6d/Singly-linked-list.svg)
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+
![Linked List](../../../assets/linked-list-overview.png)
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+
![Jest TDD Console Outputs](../../../assets/linked-list-tdd.png)
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+
![Linked List Delete Operation](../../../assets/linked-list-insert-after.png)
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+
![Linked List Delete Operation](../../../assets/linked-list-delete.png)
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.

solutions/data-structures/linked-list/LinkedList.js

+21-24
Original file line numberDiff line numberDiff line change
@@ -1,18 +1,9 @@
11
import LinkedListNode from './LinkedListNode';
2-
import Comparator from '../../utils/comparator/Comparator';
32

43
export default class LinkedList {
5-
/**
6-
* @param {Function} [comparatorFunction]
7-
*/
8-
constructor(comparatorFunction) {
9-
/** @var LinkedListNode */
4+
constructor() {
105
this.head = null;
11-
12-
/** @var LinkedListNode */
136
this.tail = null;
14-
15-
this.compare = new Comparator(comparatorFunction);
167
}
178

189
/**
@@ -66,7 +57,7 @@ export default class LinkedList {
6657
let deletedNode = null;
6758

6859
// If the head must be deleted then make 2nd node to be a head.
69-
while (this.head && this.compare.equal(this.head.value, value)) {
60+
while (this.head && this.head.value === value) {
7061
deletedNode = this.head;
7162
this.head = this.head.next;
7263
}
@@ -76,7 +67,7 @@ export default class LinkedList {
7667
if (currentNode !== null) {
7768
// If next node must be deleted then make next node to be a next next one.
7869
while (currentNode.next) {
79-
if (this.compare.equal(currentNode.next.value, value)) {
70+
if (currentNode.next.value === value) {
8071
deletedNode = currentNode.next;
8172
currentNode.next = currentNode.next.next;
8273
} else {
@@ -86,7 +77,7 @@ export default class LinkedList {
8677
}
8778

8879
// Check if tail must be deleted.
89-
if (this.compare.equal(this.tail.value, value)) {
80+
if (this.tail.value === value) {
9081
this.tail = currentNode;
9182
}
9283

@@ -96,24 +87,17 @@ export default class LinkedList {
9687
/**
9788
* @param {Object} findParams
9889
* @param {*} findParams.value
99-
* @param {function} [findParams.callback]
10090
* @return {LinkedListNode}
10191
*/
102-
find({ value = undefined, callback = undefined }) {
92+
find({ value }) {
10393
if (!this.head) {
10494
return null;
10595
}
10696

10797
let currentNode = this.head;
10898

10999
while (currentNode) {
110-
// If callback is specified then try to find node by callback.
111-
if (callback && callback(currentNode.value)) {
112-
return currentNode;
113-
}
114-
115-
// If value is specified then try to compare by value..
116-
if (value !== undefined && this.compare.equal(currentNode.value, value)) {
100+
if (value === currentNode.value) {
117101
return currentNode;
118102
}
119103

@@ -123,12 +107,24 @@ export default class LinkedList {
123107
return null;
124108
}
125109

110+
/**
111+
* @param {*} value
112+
* @return {LinkedList}
113+
*/
114+
insertAfter(value, insertValue) {
115+
const targetNode = this.find({ value: insertValue });
116+
const newNode = new LinkedListNode(value, targetNode.next);
117+
118+
targetNode.next = newNode;
119+
return newNode;
120+
}
121+
126122
/**
127123
* @return {LinkedListNode}
128124
*/
129125
deleteTail() {
126+
// There is only one node in linked list.
130127
if (this.head === this.tail) {
131-
// There is only one node in linked list.
132128
const deletedTail = this.tail;
133129
this.head = null;
134130
this.tail = null;
@@ -141,6 +137,7 @@ export default class LinkedList {
141137

142138
// Rewind to the last node and delete "next" link for the node before the last one.
143139
let currentNode = this.head;
140+
144141
while (currentNode.next) {
145142
if (!currentNode.next.next) {
146143
currentNode.next = null;
@@ -179,8 +176,8 @@ export default class LinkedList {
179176
*/
180177
toArray() {
181178
const nodes = [];
182-
183179
let currentNode = this.head;
180+
184181
while (currentNode) {
185182
nodes.push(currentNode);
186183
currentNode = currentNode.next;

0 commit comments

Comments
 (0)