Home >Backend Development >Python Tutorial >Concurrency in Python with Threading and Multiprocessing
Concurrency is a crucial idea in modern programming that allows multiple tasks to run at the same time to improve the performance of applications.
There are several ways to achieve concurrency in Python, with threading and multiprocessing being the most well-known.
In this article, we'll explore these two methods in detail, understand how they work, and discuss when to use each, along with practical code examples.
Before we talk about threading and multiprocessing, it’s important to understand what concurrency means.
Concurrency is when a program can do multiple tasks or processes at the same time.
This can make the program use resources better and run faster, especially when it needs to do things like reading files or doing lots of calculations.
There are two main ways to achieve concurrency:
Python offers two main ways to achieve concurrency:
Threading allows you to run multiple smaller units of a process, called threads, within the same process, sharing the same memory space.
Threads are lighter than processes, and switching between them is faster.
However, threading in Python is subject to the Global Interpreter Lock (GIL), which ensures only one thread can execute Python code at a time.
Python's threading module provides a simple and flexible way to create and manage threads.
Let’s start with a basic example:
import threading import time def print_numbers(): for i in range(5): print(f"Number: {i}") time.sleep(1) # Creating a thread thread = threading.Thread(target=print_numbers) # Starting the thread thread.start() # Wait for the thread to complete thread.join() print("Thread has finished executing") # Output: # Number: 0 # Number: 1 # Number: 2 # Number: 3 # Number: 4 # Thread has finished executing
In this example:
Threading is especially useful for I/O-bound tasks, such as file operations, network requests, or database queries, where the program spends most of its time waiting for external resources.
Here’s an example that simulates downloading files using threads:
import threading import time def download_file(file_name): print(f"Starting download of {file_name}...") time.sleep(2) # Simulate download time print(f"Finished downloading {file_name}") files = ["file1.zip", "file2.zip", "file3.zip"] threads = [] # Create and start threads for file in files: thread = threading.Thread(target=download_file, args=(file,)) thread.start() threads.append(thread) # Ensure all threads have finished for thread in threads: thread.join() print("All files have been downloaded.") # Output: # Starting download of file1.zip... # Starting download of file2.zip... # Starting download of file3.zip... # Finished downloading file1.zip # Finished downloading file2.zip # Finished downloading file3.zip # All files have been downloaded.
By creating and managing separate threads for each file download, the program can handle multiple tasks simultaneously, improving overall efficiency.
The key steps in the code are as follows:
While threading can improve performance for I/O-bound tasks, it has limitations:
Multiprocessing addresses the limitations of threading by using separate processes instead of threads.
Each process has its own memory space and Python interpreter, allowing true parallelism on multi-core systems.
This makes multiprocessing ideal for tasks that require heavy computation.
The multiprocessing module in Python allows you to create and manage processes easily.
Let’s start with a basic example:
import multiprocessing import time def print_numbers(): for i in range(5): print(f"Number: {i}") time.sleep(1) if __name__ == "__main__": # Creating a process process = multiprocessing.Process(target=print_numbers) # Starting the process process.start() # Wait for the process to complete process.join() print("Process has finished executing") # Output: # Number: 0 # Number: 1 # Number: 2 # Number: 3 # Number: 4 # Process has finished executing
This example is similar to the threading example, but with processes.
Notice that the process creation and management are similar to threading, but because processes run in separate memory spaces, they are truly concurrent and can run on different CPU cores.
Multiprocessing is particularly beneficial for tasks that are CPU-bound, such as numerical computations or data processing.
Here’s an example that calculates the square of numbers using multiple processes:
import multiprocessing def compute_square(number): return number * number if __name__ == "__main__": numbers = [1, 2, 3, 4, 5] # Create a pool of processes with multiprocessing.Pool() as pool: # Map function to numbers using multiple processes results = pool.map(compute_square, numbers) print("Squares:", results) # Output: # Squares: [1, 4, 9, 16, 25]
Here are the key steps in the code:
Since each process has its own memory space, sharing data between processes requires inter-process communication (IPC) mechanisms.
The multiprocessing module provides several tools for IPC, such as Queue, Pipe, and Value.
Here’s an example using Queue to share data between processes:
import multiprocessing def worker(queue): # Retrieve and process data from the queue while not queue.empty(): item = queue.get() print(f"Processing {item}") if __name__ == "__main__": queue = multiprocessing.Queue() # Add items to the queue for i in range(10): queue.put(i) # Create a pool of processes to process the queue processes = [] for _ in range(4): process = multiprocessing.Process(target=worker, args=(queue,)) processes.append(process) process.start() # Wait for all processes to complete for process in processes: process.join() print("All processes have finished.") # Output: # Processing 0 # Processing 1 # Processing 2 # Processing 3 # Processing 4 # Processing 5 # Processing 6 # Processing 7 # Processing 8 # Processing 9 # All processes have finished.
In this example:
While multiprocessing provides true parallelism, it comes with its own set of challenges:
Choosing between threading and multiprocessing depends on the type of task you're dealing with:
Use Threading:
Use Multiprocessing:
Concurrency in Python is a powerful way to make your applications run faster.
Threading is great for tasks that involve a lot of waiting, like network operations or reading/writing files, but it's not as effective for tasks that require heavy computations because of something called the Global Interpreter Lock (GIL).
On the other hand, multiprocessing allows for true parallelism, making it perfect for CPU-intensive tasks, although it comes with higher overhead and complexity.
Whether you're processing data, handling multiple network requests, or doing complex calculations, Python's threading and multiprocessing tools give you what you need to make your program as efficient and fast as possible.
The above is the detailed content of Concurrency in Python with Threading and Multiprocessing. For more information, please follow other related articles on the PHP Chinese website!