


Why are changes made to shared variables by one thread invisible to other threads in Python?
I tried using the same variable in multiple threads, but the value of the variable was not updated consistently across threads. For example, when thread 1 updates a variable to 1, thread 2 does not recognize this change and instead sees the old value.
This is a simple code example illustrating the problem. When the user presses the "a" key, the variable "query" should update and display like this:
- Query 1: a
- Query: a
However, the actual output I get is just:
- Query 1: a
Can you help me understand why this is happening and how to fix it?
import getch import threading QUERY = "" EXIT_THREAD = False def input_thread(): global EXIT_THREAD last_query = "" while not EXIT_THREAD: if last_query != QUERY: last_query = QUERY print(f"Query: {QUERY}") thread = threading.Thread(target=input_thread) thread.start() while True: char = getch.getch() if char == "\n": break elif char == "\x7f": QUERY = QUERY[:-1] else: QUERY += char print(f"Query1: {QUERY}") # kill input thread EXIT_THREAD = True thread.join()
Correct answer
The problem is that the getch
module is poorly coded. While it blocks waiting for input, it does not release the gil (global interpreter lock), so your other threads are not allowed to run. Synchronizing with the lock doesn't help much; gil has protected access to query
.
The problem is that every time you change query
, you're back to getch
, which locks the gil. When getch
returns, enough time has passed that the gil is immediately handed over, another thread checks for changes, sees the last change and reports it, and the main thread eventually gets control back, but usually after getch
Not enough is done before locking it again to cause another gil switch, and another thread never gets a chance to run and see the changes until getch
returns next time. This may vary depending on the python version (the rules for checking gil change from time to time), but it's always unstable.
The correct solution is for the getch
module to release the gil internally before making blocking calls, but failing that you can do this by deliberately prefixing each getch
with When calling gil release mode blocking to give other threads some running time, import the time
module and add a sleep to give other threads time to view the latest changes:
while True: time.sleep(0.001) # Explicitly releases GIL for a millisecond char = getch.getch()
This gets the behavior you expect, although technically you're subject to a race condition if other threads are involved, but for two threads like this it's pretty reliable.
The above is the detailed content of Why are changes made to shared variables by one thread invisible to other threads in Python?. For more information, please follow other related articles on the PHP Chinese website!

Hot AI Tools

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Undress AI Tool
Undress images for free

Clothoff.io
AI clothes remover

Video Face Swap
Swap faces in any video effortlessly with our completely free AI face swap tool!

Hot Article

Hot Tools

Notepad++7.3.1
Easy-to-use and free code editor

SublimeText3 Chinese version
Chinese version, very easy to use

Zend Studio 13.0.1
Powerful PHP integrated development environment

Dreamweaver CS6
Visual web development tools

SublimeText3 Mac version
God-level code editing software (SublimeText3)

Hot Topics



Solution to permission issues when viewing Python version in Linux terminal When you try to view Python version in Linux terminal, enter python...

When using Python's pandas library, how to copy whole columns between two DataFrames with different structures is a common problem. Suppose we have two Dats...

How to teach computer novice programming basics within 10 hours? If you only have 10 hours to teach computer novice some programming knowledge, what would you choose to teach...

How to avoid being detected when using FiddlerEverywhere for man-in-the-middle readings When you use FiddlerEverywhere...

How does Uvicorn continuously listen for HTTP requests? Uvicorn is a lightweight web server based on ASGI. One of its core functions is to listen for HTTP requests and proceed...

The article discusses popular Python libraries like NumPy, Pandas, Matplotlib, Scikit-learn, TensorFlow, Django, Flask, and Requests, detailing their uses in scientific computing, data analysis, visualization, machine learning, web development, and H

In Python, how to dynamically create an object through a string and call its methods? This is a common programming requirement, especially if it needs to be configured or run...

Using python in Linux terminal...
