-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path0430__flatten_multilevel_doubly_linked_list.py
73 lines (57 loc) · 1.94 KB
/
0430__flatten_multilevel_doubly_linked_list.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
"""
LeetCode: https://leetcode.com/problems/flatten-a-multilevel-doubly-linked-list/
"""
from dataclasses import dataclass
from typing import Optional, List
from unittest import TestCase
from lib.ListNode import flatten_list
@dataclass
class Node:
val: int = 0
prev: 'Node' = None
next: 'Node' = None
child: 'Node' = None
class Solution(TestCase):
def create_from_list(self, l: List[int]):
dummy = Node()
tail = dummy
for n in l:
node = Node(n, tail)
tail.next = node
tail = tail.next
if dummy.next:
dummy.next.prev = None
return dummy.next
def test_empty_list(self):
self.assertIsNone(self.flatten(None))
def test_just_head(self):
head = Node()
self.assertEqual(head, self.flatten(head))
def test_linear_list(self):
self.assertEqual([1, 2, 3], flatten_list(self.flatten(self.create_from_list([1, 2, 3]))))
def test_two_levels(self):
first_level_head = self.create_from_list([1, 2, 3])
second_level_head = self.create_from_list([4, 5, 6])
first_level_head.next.child = second_level_head
self.assertEqual([1, 2, 4, 5, 6, 3], flatten_list(self.flatten(first_level_head)))
def flatten(self, head: Optional[Node]) -> Optional[Node]:
if not head:
return None
dummy = Node()
tail = dummy
while head:
tail.next, head.prev = head, tail
tail = tail.next
nxt = head.next
if head.child:
# gateway to next level
next_part = self.flatten(head.child)
if next_part:
tail.next, next_part.prev = next_part, tail
while tail.next:
tail = tail.next
head.child = None
head = nxt
if dummy.next:
dummy.next.prev = None
return dummy.next