2020-11-28 05:50:18 +00:00
|
|
|
"""
|
|
|
|
Linked Lists consists of Nodes.
|
|
|
|
Nodes contain data and also may link to other nodes:
|
|
|
|
- Head Node: First node, the address of the
|
|
|
|
head node gives us access of the complete list
|
|
|
|
- Last node: points to null
|
|
|
|
"""
|
2024-03-13 06:52:41 +00:00
|
|
|
|
2021-11-17 03:43:02 +00:00
|
|
|
from __future__ import annotations
|
2020-11-28 05:50:18 +00:00
|
|
|
|
2021-11-17 03:43:02 +00:00
|
|
|
from typing import Any
|
2020-11-28 05:50:18 +00:00
|
|
|
|
|
|
|
|
2018-10-19 12:48:28 +00:00
|
|
|
class Node:
|
2022-10-13 14:23:59 +00:00
|
|
|
def __init__(self, item: Any, next: Any) -> None: # noqa: A002
|
2018-10-19 12:48:28 +00:00
|
|
|
self.item = item
|
|
|
|
self.next = next
|
|
|
|
|
2019-10-05 05:14:13 +00:00
|
|
|
|
2018-10-19 12:48:28 +00:00
|
|
|
class LinkedList:
|
2020-11-28 05:50:18 +00:00
|
|
|
def __init__(self) -> None:
|
2021-11-17 03:43:02 +00:00
|
|
|
self.head: Node | None = None
|
2020-05-31 09:37:45 +00:00
|
|
|
self.size = 0
|
2018-10-19 12:48:28 +00:00
|
|
|
|
2023-09-08 12:20:28 +00:00
|
|
|
def add(self, item: Any, position: int = 0) -> None:
|
|
|
|
"""
|
|
|
|
Add an item to the LinkedList at the specified position.
|
|
|
|
Default position is 0 (the head).
|
|
|
|
|
|
|
|
Args:
|
|
|
|
item (Any): The item to add to the LinkedList.
|
|
|
|
position (int, optional): The position at which to add the item.
|
|
|
|
Defaults to 0.
|
|
|
|
|
|
|
|
Raises:
|
|
|
|
ValueError: If the position is negative or out of bounds.
|
|
|
|
|
|
|
|
>>> linked_list = LinkedList()
|
|
|
|
>>> linked_list.add(1)
|
|
|
|
>>> linked_list.add(2)
|
|
|
|
>>> linked_list.add(3)
|
|
|
|
>>> linked_list.add(4, 2)
|
|
|
|
>>> print(linked_list)
|
|
|
|
3 --> 2 --> 4 --> 1
|
|
|
|
|
|
|
|
# Test adding to a negative position
|
|
|
|
>>> linked_list.add(5, -3)
|
|
|
|
Traceback (most recent call last):
|
|
|
|
...
|
|
|
|
ValueError: Position must be non-negative
|
|
|
|
|
|
|
|
# Test adding to an out-of-bounds position
|
|
|
|
>>> linked_list.add(5,7)
|
|
|
|
Traceback (most recent call last):
|
|
|
|
...
|
|
|
|
ValueError: Out of bounds
|
|
|
|
>>> linked_list.add(5, 4)
|
|
|
|
>>> print(linked_list)
|
|
|
|
3 --> 2 --> 4 --> 1 --> 5
|
|
|
|
"""
|
|
|
|
if position < 0:
|
|
|
|
raise ValueError("Position must be non-negative")
|
|
|
|
|
|
|
|
if position == 0 or self.head is None:
|
|
|
|
new_node = Node(item, self.head)
|
|
|
|
self.head = new_node
|
|
|
|
else:
|
|
|
|
current = self.head
|
|
|
|
for _ in range(position - 1):
|
|
|
|
current = current.next
|
|
|
|
if current is None:
|
|
|
|
raise ValueError("Out of bounds")
|
|
|
|
new_node = Node(item, current.next)
|
|
|
|
current.next = new_node
|
2020-05-31 09:37:45 +00:00
|
|
|
self.size += 1
|
2018-10-19 12:48:28 +00:00
|
|
|
|
2020-11-28 05:50:18 +00:00
|
|
|
def remove(self) -> Any:
|
2021-10-07 15:18:23 +00:00
|
|
|
# Switched 'self.is_empty()' to 'self.head is None'
|
|
|
|
# because mypy was considering the possibility that 'self.head'
|
|
|
|
# can be None in below else part and giving error
|
|
|
|
if self.head is None:
|
2018-10-19 12:48:28 +00:00
|
|
|
return None
|
|
|
|
else:
|
|
|
|
item = self.head.item
|
|
|
|
self.head = self.head.next
|
2020-05-31 09:37:45 +00:00
|
|
|
self.size -= 1
|
2018-10-19 12:48:28 +00:00
|
|
|
return item
|
|
|
|
|
2020-11-28 05:50:18 +00:00
|
|
|
def is_empty(self) -> bool:
|
2018-10-19 12:48:28 +00:00
|
|
|
return self.head is None
|
2020-05-31 09:37:45 +00:00
|
|
|
|
2020-11-28 05:50:18 +00:00
|
|
|
def __str__(self) -> str:
|
|
|
|
"""
|
|
|
|
>>> linked_list = LinkedList()
|
|
|
|
>>> linked_list.add(23)
|
|
|
|
>>> linked_list.add(14)
|
|
|
|
>>> linked_list.add(9)
|
|
|
|
>>> print(linked_list)
|
|
|
|
9 --> 14 --> 23
|
|
|
|
"""
|
2022-11-15 17:29:14 +00:00
|
|
|
if self.is_empty():
|
2020-11-28 05:50:18 +00:00
|
|
|
return ""
|
|
|
|
else:
|
|
|
|
iterate = self.head
|
|
|
|
item_str = ""
|
2021-10-07 15:18:23 +00:00
|
|
|
item_list: list[str] = []
|
2020-11-28 05:50:18 +00:00
|
|
|
while iterate:
|
|
|
|
item_list.append(str(iterate.item))
|
|
|
|
iterate = iterate.next
|
|
|
|
|
|
|
|
item_str = " --> ".join(item_list)
|
|
|
|
|
|
|
|
return item_str
|
|
|
|
|
|
|
|
def __len__(self) -> int:
|
2020-05-31 09:37:45 +00:00
|
|
|
"""
|
|
|
|
>>> linked_list = LinkedList()
|
|
|
|
>>> len(linked_list)
|
|
|
|
0
|
|
|
|
>>> linked_list.add("a")
|
|
|
|
>>> len(linked_list)
|
|
|
|
1
|
|
|
|
>>> linked_list.add("b")
|
|
|
|
>>> len(linked_list)
|
|
|
|
2
|
|
|
|
>>> _ = linked_list.remove()
|
|
|
|
>>> len(linked_list)
|
|
|
|
1
|
|
|
|
>>> _ = linked_list.remove()
|
|
|
|
>>> len(linked_list)
|
|
|
|
0
|
|
|
|
"""
|
|
|
|
return self.size
|