Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions python-deque/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
# Python's deque: Implement Efficient Queues and Stacks

This folder provides the code examples for the Real Python tutorial [Python's deque: Implement Efficient Queues and Stacks](https://realpython.com/python-deque/).
43 changes: 43 additions & 0 deletions python-deque/custom_queue.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
from collections import deque


class Queue:
def __init__(self):
self._items = deque()

def enqueue(self, item):
self._items.append(item)

def dequeue(self):
try:
return self._items.popleft()
except IndexError:
# Break exception chain to hide implementation details
raise IndexError("dequeue from an empty queue") from None

def __len__(self):
return len(self._items)

def __contains__(self, item):
return item in self._items

def __iter__(self):
yield from self._items

def __reversed__(self):
yield from reversed(self._items)

def __repr__(self):
return f"Queue({list(self._items)})"


if __name__ == "__main__":
queue = Queue()
print("Enqueueing items with '.enqueue()'...")
queue.enqueue(1)
queue.enqueue(2)
queue.enqueue(3)
print(queue)
print("Dequeueing one item with '.dequeue()'...")
print(queue.dequeue())
print(queue)
18 changes: 18 additions & 0 deletions python-deque/page_history.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
from collections import deque

page_history = deque(maxlen=3)

urls = ("https://google.com", "https://yahoo.com", "https://www.bing.com")

for url in urls:
page_history.appendleft(url)

print(page_history)

page_history.appendleft("https://youtube.com")

print(page_history)

page_history.appendleft("https://facebook.com")

print(page_history)
42 changes: 42 additions & 0 deletions python-deque/producer_consumer.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
import logging
import random
import threading
import time
from collections import deque

logging.basicConfig(level=logging.INFO, format="%(message)s")


def _wait_seconds(mins, maxs):
time.sleep(mins + random.random() * (maxs - mins))


def produce(queue, size):
while True:
if len(queue) < size:
value = random.randint(0, 9)
queue.append(value)
logging.info("Produced: %d -> %s", value, str(queue))
else:
logging.info("Queue is saturated")
_wait_seconds(0.1, 0.5)


def consume(queue):
while True:
try:
value = queue.popleft()
except IndexError:
logging.info("Queue is empty")
else:
logging.info("Consumed: %d -> %s", value, str(queue))
_wait_seconds(0.2, 0.7)


logging.info("Starting Threads...\n")
logging.info("Press Ctrl+C to interrupt the execution\n")

shared_queue = deque()

threading.Thread(target=produce, args=(shared_queue, 10)).start()
threading.Thread(target=consume, args=(shared_queue,)).start()
13 changes: 13 additions & 0 deletions python-deque/tail.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
from collections import deque


def tail(filename, lines=10):
try:
with open(filename) as file:
return deque(file, maxlen=lines)
except OSError as error:
print(f'Opening file "{filename}" failed with error: {error}')


if __name__ == "__main__":
print(tail("./README.md"))
25 changes: 25 additions & 0 deletions python-deque/time_append.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
from collections import deque
from time import perf_counter

TIMES = 10_000
NANOSECONDS_PER_SECOND = 1e9
a_list = []
a_deque = deque()


def average_time(func, times):
total = 0.0
for i in range(times):
start = perf_counter()
func(i)
# Convert to ns to improve readability
total += (perf_counter() - start) * NANOSECONDS_PER_SECOND
return total / times


list_time = average_time(lambda i: a_list.insert(0, i), TIMES)
deque_time = average_time(lambda i: a_deque.appendleft(i), TIMES)
gain = list_time / deque_time

print(f"list.insert() {list_time:.6} ns")
print(f"deque.appendleft() {deque_time:.6} ns ({gain:.6}x faster)")
25 changes: 25 additions & 0 deletions python-deque/time_pop.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
from collections import deque
from time import perf_counter

TIMES = 10_000
NANOSECONDS_PER_SECOND = 1e9
a_list = [1] * TIMES
a_deque = deque(a_list)


def average_time(func, times):
total = 0.0
for _ in range(times):
start = perf_counter()
func()
# Convert to ns to improve readability
total += (perf_counter() - start) * NANOSECONDS_PER_SECOND
return total / times


list_time = average_time(lambda: a_list.pop(0), TIMES)
deque_time = average_time(lambda: a_deque.popleft(), TIMES)
gain = list_time / deque_time

print(f"list.pop(0) {list_time:.6} ns")
print(f"deque.popleft() {deque_time:.6} ns ({gain:.6}x faster)")
33 changes: 33 additions & 0 deletions python-deque/time_random_access.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
from collections import deque
from time import perf_counter

TIMES = 10_000
MICROSECONDS_PER_SECOND = 1e6
a_list = [1] * TIMES
a_deque = deque(a_list)


def average_time(func, times):
total = 0.0
for _ in range(times):
start = perf_counter()
func()
# Convert to μs to improve readability
total += (perf_counter() - start) * MICROSECONDS_PER_SECOND
return total / times


def time_it(sequence):
middle = len(sequence) // 2
sequence.insert(middle, "middle")
sequence[middle]
sequence.remove("middle")
del sequence[middle]


list_time = average_time(lambda: time_it(a_list), TIMES)
deque_time = average_time(lambda: time_it(a_deque), TIMES)
gain = deque_time / list_time

print(f"list {list_time:.6} μs ({gain:.6}x faster)")
print(f"deque {deque_time:.6} μs")