Problem Description

You are given the head of a singly linked-list. The list can be represented as:

$L_0 → L_1 → … → L_{(n - 1)} → L_n$

Reorder the list to be on the following form:

$L_0 → L_n → L_1 → L_{(n - 1)} → L_2 → L_{(n - 2)} → …$

You may not modify the values in the list’s nodes. Only nodes themselves may be changed.

Example 1:

143.1.jpg

  • Input: head = [1,2,3,4]
  • Output: [1,4,2,3]

Example 2:

143.2.jpg

  • Input: head = [1,2,3,4,5]
  • Output: [1,5,2,4,3]

My Idea

One of the challenges here is to modify head in-place. This led me to the following approach: We separate the list in half by using two pointers a slow and a fast one, where the slow crawls 1 node at a time and the fast one 2. This means that if we have an even number of nodes once the fast one reaches the last element the slow one points at the last element from the first half. If we have an uneven number of nodes, once the fast one reaches the end, the slow one is pointing at the middle element, which for our purpose is also the last one from the first half. Then what we do is utilizing the logic from problem 206. Reverse Linked List we reverse the second half of the list. After which we merge the two halves. This yields a time complexity of O(n).

My solution

from typing import Optional
# Definition for singly-linked list.
class ListNode:
    def __init__(self, val=0, next=None):
        self.val = val
        self.next = next
# Time Complexity: O(n)
def reoderList(head: Optional[ListNode]) -> None:
    # Do not return anything, modify head in-place instead!
    l = head
    # Separate the list in two halves
    r = head
    f = head.next
    while f and f.next:
        r = r.next
        f = f.next.next
    tmp = r.next
    r.next = None
    r = tmp
    # Reverse second half
    prev = None
    curr = r
    while curr:
        next = curr.next
        curr.next = prev
        prev = curr
        curr = next
    r = prev
    # Merge both halves
    while l and r:
        tl = l.next
        tr = r.next
        l.next = r
        l.next.next=tl
        r = tr
        l = l.next.next