Concurrent Programming
One of Rust's major goals as a language is to enable fearless concurrency. So much so that an entire chapter of the Rust Book is devoted to it.
In Python, concurrency is possible however we are impacted by the GIL.
What's really fascinating (to me, anyways) is how decisions about how memory is managed in both languages has a direct impact on how concurrency is handled.
Before we dig into concurrency, let's take a step back and talk about memory.
Memory
Every programming language stores objects in memory. Whether it's variables, functions, or other data, we store these in memory to allow fast access to them when we need them.
How languages manage memory defines the flavor and performance characteristics of the language.
The GIL and Python's Memory Management
In Python, the infamous Global Interpreter Lock (or GIL) exists because objects in Python are reference counted. This means that every object has a counter associated with it that is incremented as it is referenced and decremented as it is removed from scope. When an object has 0 references, it is cleared from memory, freeing up space.
To prevent two different threads from accessing or releasing the same reference to an object, the GIL is used to prevent multiple threads from accessing the same object. This has the effect of serializing access to objects in memory and effectively making CPU-bound Python code single-threaded.
To work around these limitations, CPU-bound Python code has to rely on threads, which has its own set of limitations and overhead costs.