1669. Merge In Between Linked Lists - Detailed Explanation

Free Coding Questions Catalog
Boost your coding skills with our essential coding questions catalog. Take a step towards a better tech career now!

Problem Statement

You are given two linked lists: list1 and list2. Also, you are given two integers, a and b. Your task is to remove the nodes in list1 from index a to b (inclusive) and then insert list2 in their place. Finally, return the head of the modified list.

Example Inputs and Outputs

Example 1:

  • Input:
    • list1: 0 → 1 → 2 → 3 → 4 → 5
    • list2: 1000000 → 1000001 → 1000002
    • a = 3, b = 4
  • Output: 0 → 1 → 2 → 1000000 → 1000001 → 1000002 → 5
  • Explanation:
    Nodes 3 and 4 in list1 (at indices 3 and 4) are removed. list2 is inserted between node 2 and node 5.

Example 2:

  • Input:
    • list1: 0 → 1 → 2 → 3 → 4 → 5 → 6
    • list2: 1000000 → 1000001 → 1000002 → 1000003 → 1000004 → 1000005
    • a = 2, b = 5
  • Output: 0 → 1 → 1000000 → 1000001 → 1000002 → 1000003 → 1000004 → 1000005 → 6
  • Explanation:
    Nodes 2, 3, 4, and 5 (indices 2 through 5) are removed from list1. list2 is inserted between node 1 and node 6.

Example 3:

  • Input:
    • list1: 5 → 6 → 7 → 8 → 9
    • list2: 1 → 2 → 3
    • a = 1, b = 2
  • Output: 5 → 1 → 2 → 3 → 8 → 9
  • Explanation:
    Nodes 6 and 7 (indices 1 and 2) in list1 are removed and list2 is inserted between node 5 and node 8.

Constraints:

  • The indices a and b will always be valid (0-indexed) and a <= b.
  • list1 will have at least b + 1 nodes.
  • list2 is a non-empty linked list.
  • The problem assumes you have basic operations on linked lists (traversal, pointer manipulation).

Hints Before the Solution

  1. Find Key Pointers in list1:

    • Traverse list1 to find the node immediately before the ath node (let’s call it nodeA_prev).
    • Also, find the node at index b (let’s call it nodeB). You will need the node right after nodeB to connect to list2 later.
  2. Find the Tail of list2:

    • Traverse list2 to find its last node (the tail). This tail will eventually be connected to the node that comes after nodeB in list1.
  3. Connect the Lists:

    • Set nodeA_prev.next to the head of list2.
    • Then, set the tail of list2.next to the node after nodeB.

Approaches

Approach 1: Brute Force Rebuild

In the brute force approach, instead of modifying the original list in place, you build a new linked list from scratch by copying over the nodes you want to keep and then inserting the nodes from the second list. This approach is straightforward conceptually, especially if you’re not comfortable with pointer manipulations.

Steps:

  1. Traverse and Copy First Part of list1:

    • Start from the head of list1.
    • Traverse until you reach index a - 1 (i.e., the node just before the removal section).
    • For each node up to index a - 1, create a copy (or just reassign if allowed) and append it to your new list.
  2. Insert All Nodes from list2:

    • Traverse list2 from its head.
    • Append each node from list2 to the new list.
    • Note: This requires iterating through the entire list2 to get all its nodes.
  3. Traverse and Copy the Remaining Part of list1:

    • Continue from index b + 1 in list1.
    • Copy all remaining nodes from list1 into the new list.
  4. Return the Head of the New List:

    • The new list now represents the merged result with the nodes from index a to b removed and list2 inserted in their place.

Pros and Cons:

  • Pros:

    • Simplicity: The logic of building a new list is straightforward.
    • Clarity: You avoid complex pointer manipulations that can lead to bugs.
  • Cons:

    • Extra Space: You use additional space for the new list (even if you reuse node values).
    • Inefficiency: The need to traverse parts of list1 twice (once to copy the beginning and once for the ending) and the whole of list2 increases time overhead compared to a direct pointer reattachment.

When to Use This Approach:

  • When you want to prioritize code clarity over efficiency.
  • In environments where extra space isn’t a concern, or when working in languages/environments that make pointer manipulations error-prone.

Approach 2: Optimal In‑Place Merge

This approach focuses on adjusting the pointers in the original lists rather than building a new list. The idea is to splice list2 directly into list1 by removing a section of list1 and reconnecting the nodes without creating new nodes or using extra memory.

Detailed Steps:

  1. Find the Node Before the Removal Section (nodeA_prev):

    • Traverse list1 from the head.
    • Stop when you reach the node at index a - 1 (this node will serve as the connecting point for list2).
    • Visualization:
      If list1 is 0 → 1 → 2 → 3 → 4 → 5 and a = 3, then nodeA_prev is the node with value 2.
  2. Locate the End of the Removal Section (nodeB):

    • Continue from nodeA_prev (or start a fresh traversal) until you reach the node at index b.
    • This node marks the end of the section that will be removed.
    • Visualization:
      For the same list1 with b = 4, nodeB is the node with value 4.
  3. Find the Tail of list2:

    • Traverse list2 to get to its last node.
    • This tail will later be used to connect back to list1.
    • Visualization:
      If list2 is 1000000 → 1000001 → 1000002, then the tail is the node with value 1000002.
  4. Reconnect Pointers to Merge the Lists:

    • Linking the Start:
      Set the next pointer of nodeA_prev to point to the head of list2.
    • Linking the End:
      Set the next pointer of the tail of list2 to the node that comes immediately after nodeB in list1 (i.e., nodeB.next).
    • Result:
      This effectively removes the nodes from index a to b in list1 and inserts list2 in their place without using extra space.
  5. Return the Modified list1:

    • The head of list1 remains unchanged (unless a is 0, which would require updating the head).

Pros and Cons:

  • Pros:

    • Space Efficient: Uses only a few pointers without allocating extra memory.
    • Time Efficient: Each list is traversed only as much as needed.
    • Direct: Directly manipulates the original data structure, which is often required in interview settings.
  • Cons:

    • Pointer Complexity: Requires careful pointer manipulation. A small mistake can lead to memory leaks or broken links between nodes.
    • Edge Cases: Special cases (e.g., when a is 0) must be handled explicitly, which may complicate the logic.

When to Use This Approach:

  • In interviews and real-world scenarios where space and time efficiency are paramount.
  • When you are comfortable with linked list pointer manipulations and want to optimize for performance.

Detailed Walkthrough and Visual Example (Optimal Approach)

Step-by-Step Process:

  1. Identify the Break Points:

    • For list1:
      • nodeA_prev: The node just before index a.
      • nodeB: The node at index b.
    • For list2:
      • list2_tail: The last node in list2.
  2. Visual Example (Using Example 1):

    list1: 0 → 1 → 2 → [3 → 4] → 5
                   ^         ^
                   |         |
              nodeA_prev   nodeB
    
    list2: 1000000 → 1000001 → 1000002
                    ^                   ^
                    |                   |
                list2_head         list2_tail
    

    Reconnection:

    • Set nodeA_prev.next = list2_head.
    • Set list2_tail.next = nodeB.next (which is node 5 in list1).

    Resulting list:

    0 → 1 → 2 → 1000000 → 1000001 → 1000002 → 5
    
  3. Pointer Reconnection:

    • This connects the untouched part of list1 with the whole of list2, and then reattaches the tail of list2 with the remainder of list1.

Code Implementations

Python Code

Python3
Python3

. . . .

Java Code

Java
Java

. . . .

Complexity Analysis

  • Time Complexity:

    • Traversing list1 up to index b is O(N1) where N1 is the length of list1.
    • Traversing list2 to find its tail is O(N2) where N2 is the length of list2.
    • Overall: O(N1 + N2).
  • Space Complexity:

    • Only a few pointers are used, so O(1) extra space.

Step-by-Step Walkthrough and Visual Examples

  1. Finding the Breakpoints:

    • For list1 = 0 → 1 → 2 → 3 → 4 → 5, a = 3, b = 4:
      • nodeA_prev: Node with value 2 (index 2).
      • nodeB: Node with value 4 (index 4).
  2. Merging Process:

    • Set nodeA_prev.next (node with value 2) to the head of list2 (1000000).
    • Find the tail of list2 (node with value 1000002).
    • Set tail.next to node after nodeB (node with value 5).
  3. Visual Diagram:

    Before Merge:
    list1: 0 → 1 → 2 → 3 → 4 → 5
    list2: 1000000 → 1000001 → 1000002
    
    After Reconnecting:
    0 → 1 → 2
               \
                1000000 → 1000001 → 1000002
                                       \
                                        5
    
  4. Final List:
    The final merged list is:
    0 → 1 → 2 → 1000000 → 1000001 → 1000002 → 5

Common Mistakes & Edge Cases

  • Common Mistakes:

    • Incorrect Indexing: Failing to correctly identify the node before index a or the node at index b.
    • Null Pointer Errors: Not checking if the list2 is non-empty or if nodeB.next exists.
    • Breaking the List: Forgetting to disconnect the removed sub-list from list1 which can lead to cycles or memory leaks in languages with manual memory management.
  • Edge Cases:

    • When a is 0 (i.e., the removal starts at the head).
      • In such cases, the head of list1 would change. (May require additional handling.)
    • When list2 has only one node.
    • When list1 has exactly b + 1 nodes (i.e., the removed part is at the end).
  • Variations:
    • Merging Two Sorted Linked Lists: Combining two sorted lists into one sorted list.
    • Inserting a Node in a Sorted Linked List: Finding the correct position to insert a node.
    • Deleting a Subsection of a Linked List: Removing a continuous segment from a linked list.
  • LeetCode 23: Merge k Sorted Lists - A harder problem that extends the two-list merge concept to multiple lists.

  • LeetCode 21: Merge Two Sorted Lists - An essential problem for practicing merging linked lists.

  • LeetCode 19: Remove Nth Node From End of List - Involves pointer manipulation and handling edge cases.

  • LeetCode 148: Sort List - Requires a deep understanding of linked list manipulations and often uses merge sort.

TAGS
LeetCode
CONTRIBUTOR
Design Gurus Team
-

GET YOUR FREE

Coding Questions Catalog

Design Gurus Newsletter - Latest from our Blog
Boost your coding skills with our essential coding questions catalog.
Take a step towards a better tech career now!
Explore Answers
Structured approaches to break down unfamiliar coding tasks
What is HTTP Long Polling?
Simulating interviewer interruptions to maintain composure
Related Courses
Image
Grokking the Coding Interview: Patterns for Coding Questions
Grokking the Coding Interview Patterns in Java, Python, JS, C++, C#, and Go. The most comprehensive course with 476 Lessons.
Image
Grokking Modern AI Fundamentals
Master the fundamentals of AI today to lead the tech revolution of tomorrow.
Image
Grokking Data Structures & Algorithms for Coding Interviews
Unlock Coding Interview Success: Dive Deep into Data Structures and Algorithms.
Image
One-Stop Portal For Tech Interviews.
Copyright © 2025 Design Gurus, LLC. All rights reserved.
;