Good morning! Here's our prompt for today.
You're sent a linked list of numbers, but it's been received in the opposite order to what you need. This has happened multiple times now, so you decide to write an algorithm to reverse the lists as they come in. The list you've received is as follows:
17 -> 2 -> 21 -> 6 -> 42 -> 10Write an algorithm for a method reverseList that takes in a head node as a parameter, and reverses the linked list. It should be capable of reversing a list of any length.

You may use the example linked list for testing purposes. Your method will be called as such:
1class LinkedListNode {
2 constructor(val, next = null) {
3 this.val = val;
4 this.next = next;
5 }
6}
7
8l1 = new LinkedListNode(1);
9l1.next = new LinkedListNode(2);
10reverseList(l1);Constraints
- Length of the given LinkedList <=
100000 - The nodes will always contain integer values between
-1000000000and1000000000 - Expected time complexity :
O(n) - Expected space complexity :
O(1)
Try to solve this here or in Interactive Mode.
How do I practice this challenge?
xxxxxxxxxx 'PASSED: Reversing linked list `3 -> 4 -> 5 -> 6 -> 7 -> 8 -> 9 -> 10` should return `10 -> 9 -> 8 -> 7 -> 6 -> 5 -> 4 -> 3`'var assert = require('assert');​function reverseList(head) { // Fill in this method return head;}​function Node(val) { this.val = val; this.next = null;}​function LinkedListNode(val) { this.val = val; this.next = null;}​var list1 = new LinkedListNode(3);var nodes1 = [4, 5, 6, 7, 8, 9, 10];createNodes(list1, nodes1);​var list2 = new LinkedListNode(1);var nodes2 = [2, 3, 4, 5, 6, 7, 8];createNodes(list2, nodes2);​function createNodes(head, nodes) { for (let i = 0; i < nodes.length; i++) { var newNode = new LinkedListNode(nodes[i]); head.next = newNode;Here's our guided, illustrated walk-through.
How do I use this guide?
Let's take the most basic linked list first-- one with two nodes. We need to reverse this.
Seems pretty easy, right? To reverse an entire linked list, simply reverse every pointer. If 1 is pointing at 2, flip it so 2 should point to 1.
Of course, there's complexities that lie ahead. But for now, that's the basis for:
117 -> 2 -> 21 -> 6 -> 42 -> 10becoming
117 <- 2 <- 21 <- 6 <- 42 <- 10The actual reversal method is actually pretty straightforward, but be aware that it takes some time to reason out. It's easy to get lost, so make sure you draw lots of diagrams.
As this is a problem (reversing an entire linked list) that can be broken up into sub-problems (reverse the pointer between just two nodes), it seems like a good opportunity to use recursion.
There are many ways to do the actual reversal, and we'll cover both an iterative and recursive approach, but the general methodology is as follows:
- Begin by creating 3 pointers:
newHead,headandnextNode.newHeadandnextNodeare initialized tonull.headstarts off pointing to the head of the linked list.
- Iterate (or recursively do) through the following process until
headisnull. This means that the end of the list has been reached:
xxxxxxxxxxclass LinkedListNode { constructor(val, next = null) { this.val = val; this.next = next; }}​l1 = new LinkedListNode(1);l2 = new LinkedListNode(2);l1.next = l2;​// we start at headlet head = l1;let newHead = null;while (head != null) { // store the node to the right to reuse later let nextNode = head.next; // set the current node's next to point backwards head.next = newHead; // store the current node, to be used as the new next later newHead = head; // the previously right-side node is now processed head = nextNode;}​console.log(l2);It's difficult to visualize this chain of events, so let's use comments to visualize it. During the interview, try not to keep it in your head.
Here's a pro-tip: many interviews are conducted online now, so you'll be able to type out things. You can use ASCII-esque linked lists like 1 -> 2 -> 3 to keep the directionality clear. If you're still on a whiteboard, diagram away!
It'll be especially difficult while balancing your nerves and talking to the interviewer. Take advantage of the whiteboard not just to record things, but also to think through potential steps.
Let's walk through it step by step and then look at working code. Let's reverse an extremely basic list, like 8 -> 4. The first line is let nextNode = head.next;, which will store the node to the right.
xxxxxxxxxxnextNode = 4// 8 -> 4Then we'll do head.next = newHead;, which will set the current node's next to point backwards.
xxxxxxxxxxnextNode = 4// <- 8, 4Now newHead = head; will store the current node, to be used as the new next later.
xxxxxxxxxxnewHead = 8nextNode = 4// <- 8, 4Finally, the previously right-side node is now processed:
xxxxxxxxxxnewHead = 8nextNode = 4// <- 8, 4 ^ current nodeNow we process the next one with the same steps. nextNode = head.next; will store the node to the right.
xxxxxxxxxxnewHead = 8nextNode = null// <- 8, 4 ^ current nodeAgain, set the current node's next to point backwards with head.next = newHead;. Recall that newHead is 8! This is where we make the switch:
xxxxxxxxxxnewHead = 8nextNode = null// <- 8 <- 4 ^ current nodeNow let's see this all put together in code, with lots of comments for edification!
xxxxxxxxxxconsole.log(l2);class LinkedListNode { constructor(val, next = null) { this.val = val; this.next = next; }}​l1 = new LinkedListNode(8);l2 = new LinkedListNode(4);l1.next = l2;​// start at head, 8let head = l1;// example: 8 -> 4let newHead = null;while (head) { /* FIRST PASS */ // store the node to the right let nextNode = head.next; // nextNode = 4, still 8 -> 4 // set the current node's next to point backwards head.next = newHead; // 8 -> null // store the current node, to be used as the new next later newHead = head; // newHead = 8 // the previously right-side node is now processed head = nextNode; // head = 4Does that all make sense? Be sure to go through the iterative approach a few times.
Here's the recursive way to do it. This can also be tricky, especially on first glance, but realize most of the magic happens when it gets to the end.
xxxxxxxxxxclass LinkedListNode { constructor(val, next = null) { this.val = val; this.next = next; }}​l1 = new LinkedListNode(8);l2 = new LinkedListNode(4);l1.next = l2;​function reverseList(head) { if (!head || !head.next) { return head; }​ let rest = reverseList(head.next);​ head.next.next = head; delete head.next; return rest;}​console.log(reverseList(l1));Let's take an easy example of 8 -> 4 again. let rest = reverseList(head.next); takes 4 and calls reverseList on it.
Calling reverseList on 4 will have us reach the termination clause because there is no .next:
1if (!head || !head.next) {
2 return head;
3}We go up the stack back to when 8 was being processed. rest now simply points to 4. Now notice what happens:
1// remember, head is 8 - it is being processed
2// head.next is 4
3head.next.next = head;
4// head.next.next was null since 4 wasn't pointing to anything
5// but now head.next (4) points to 8And we return 4 - which is pointing to 8. And we can simply extrapolate that to longer linked lists! Note that the recursive approach requires more space because we need to maintain our call stack.
Complexity of Final Solution
Let n be the number of nodes in the linked list. We traverse through all n nodes for O(n) linear time complexity. We reuse the old linked list nodes, so our space complexity is a constant O(1) as we do not allocate any extra space.
One Pager Cheat Sheet
- You need to write a
reverseListalgorithm that takes in aheadnode as a parameter and reverses a linked list of any length inO(n)time withO(1)space. - To reverse a linked list, simply
flipeach pointer, so that thenextreferences thepreviousnode. - It takes time and effort to properly
reversea linked list, so be sure to userecursionand draw diagrams to help keep your progress on track. - The general methodology for reversing a linked list involves a combination of
iterativeandrecursivetechniques that involve creating and manipulating 3 pointers:newHead,headandnextNode. - Keep track of the interview flow by using online comments or, if on a whiteboard,
diagramsto help visualize the sequence of events. Take advantage of the whiteboard to think through potential steps for reversing antechnical termlist, and then look at working code.- We will change the node's
nextreference to point to the previous node, usinghead.next = newHead;. - Storing the current node in
newHeadwill be used as thenextlater. - The right-side node is now
processedand has become the previous node. - We
store the node to the rightby settingnextNode = head.next;and repeat the same process with the next node. - We make the switch by setting the current node's
nexttonewHead, which is8. - We can
Seehow all the key concepts are put together in code, with lots of explaining comments! - The recursive approach to this can be tricky, especially on first glance, but most of the
magichappens when it reaches the end. - The
reverseListfunction is called on4, which reaches the termination clause due to there being no.next. 4was originally not pointing to anything, but afterhead.next.next = headit now points back to8.- We have achieved
O(n)linear time complexity andO(1)space complexity when traversing a linked list and reusing the old nodes.
This is our final solution.
To visualize the solution and step through the below code, click Visualize the Solution on the right-side menu or the VISUALIZE button in Interactive Mode.
xxxxxxxxxx}var assert = require('assert');​// iterativefunction reverseList(head) { if (head.length < 2) { return; }​ let newHead = null; while (head != null) { let nextNode = head.next; head.next = newHead; newHead = head; head = nextNode; } return newHead;}​// recursivefunction reverseList(head) { if (!head || !head.next) { return head; }​ let rest = reverseList(head.next);​ head.next.next = head; delete head.next; return rest;}​function Node(val) {That's all we've got! Let's move on to the next tutorial.
If you had any problems with this tutorial, check out the main forum thread here.


