
- Python 基礎
- Python - 首頁
- Python - 概述
- Python - 歷史
- Python - 特性
- Python vs C++
- Python - Hello World 程式
- Python - 應用領域
- Python - 直譯器
- Python - 環境搭建
- Python - 虛擬環境
- Python - 基本語法
- Python - 變數
- Python - 資料型別
- Python - 型別轉換
- Python - Unicode 系統
- Python - 字面量
- Python - 運算子
- Python - 算術運算子
- Python - 比較運算子
- Python - 賦值運算子
- Python - 邏輯運算子
- Python - 位運算子
- Python - 成員運算子
- Python - 身份運算子
- Python - 運算子優先順序
- Python - 註釋
- Python - 使用者輸入
- Python - 數字
- Python - 布林值
- Python 控制語句
- Python - 控制流
- Python - 決策
- Python - if 語句
- Python - if else
- Python - 巢狀 if
- Python - Match-Case 語句
- Python - 迴圈
- Python - for 迴圈
- Python - for-else 迴圈
- Python - while 迴圈
- Python - break 語句
- Python - continue 語句
- Python - pass 語句
- Python - 巢狀迴圈
- Python 函式 & 模組
- Python - 函式
- Python - 預設引數
- Python - 關鍵字引數
- Python - 僅限關鍵字引數
- Python - 位置引數
- Python - 僅限位置引數
- Python - 可變引數
- Python - 變數作用域
- Python - 函式註解
- Python - 模組
- Python - 內建函式
- Python 字串
- Python - 字串
- Python - 字串切片
- Python - 修改字串
- Python - 字串連線
- Python - 字串格式化
- Python - 跳脫字元
- Python - 字串方法
- Python - 字串練習
- Python 列表
- Python - 列表
- Python - 訪問列表元素
- Python - 修改列表元素
- Python - 新增列表元素
- Python - 刪除列表元素
- Python - 遍歷列表
- Python - 列表推導式
- Python - 排序列表
- Python - 複製列表
- Python - 合併列表
- Python - 列表方法
- Python - 列表練習
- Python 元組
- Python - 元組
- Python - 訪問元組元素
- Python - 更新元組
- Python - 解包元組
- Python - 遍歷元組
- Python - 合併元組
- Python - 元組方法
- Python - 元組練習
- Python 集合
- Python - 集合
- Python - 訪問集合元素
- Python - 新增集合元素
- Python - 刪除集合元素
- Python - 遍歷集合
- Python - 合併集合
- Python - 複製集合
- Python - 集合運算子
- Python - 集合方法
- Python - 集合練習
- Python 字典
- Python - 字典
- Python - 訪問字典元素
- Python - 修改字典元素
- Python - 新增字典元素
- Python - 刪除字典元素
- Python - 字典檢視物件
- Python - 遍歷字典
- Python - 複製字典
- Python - 巢狀字典
- Python - 字典方法
- Python - 字典練習
- Python 陣列
- Python - 陣列
- Python - 訪問陣列元素
- Python - 新增陣列元素
- Python - 刪除陣列元素
- Python - 遍歷陣列
- Python - 複製陣列
- Python - 反轉陣列
- Python - 排序陣列
- Python - 合併陣列
- Python - 陣列方法
- Python - 陣列練習
- Python 檔案處理
- Python - 檔案處理
- Python - 寫入檔案
- Python - 讀取檔案
- Python - 重新命名和刪除檔案
- Python - 目錄
- Python - 檔案方法
- Python - OS 檔案/目錄方法
- Python - OS 路徑方法
- 面向物件程式設計
- Python - OOPs 概念
- Python - 類 & 物件
- Python - 類屬性
- Python - 類方法
- Python - 靜態方法
- Python - 建構函式
- Python - 訪問修飾符
- Python - 繼承
- Python - 多型
- Python - 方法重寫
- Python - 方法過載
- Python - 動態繫結
- Python - 動態型別
- Python - 抽象
- Python - 封裝
- Python - 介面
- Python - 包
- Python - 內部類
- Python - 匿名類和物件
- Python - 單例類
- Python - 包裝器類
- Python - 列舉
- Python - 反射
- Python 錯誤 & 異常
- Python - 語法錯誤
- Python - 異常
- Python - try-except 塊
- Python - try-finally 塊
- Python - 丟擲異常
- Python - 異常鏈
- Python - 巢狀 try 塊
- Python - 使用者自定義異常
- Python - 日誌記錄
- Python - 斷言
- Python - 內建異常
- Python 多執行緒
- Python - 多執行緒
- Python - 執行緒生命週期
- Python - 建立執行緒
- Python - 啟動執行緒
- Python - 執行緒連線
- Python - 執行緒命名
- Python - 執行緒排程
- Python - 執行緒池
- Python - 主執行緒
- Python - 執行緒優先順序
- Python - 守護執行緒
- Python - 執行緒同步
- Python 同步
- Python - 執行緒間通訊
- Python - 執行緒死鎖
- Python - 中斷執行緒
- Python 網路程式設計
- Python - 網路程式設計
- Python - Socket 程式設計
- Python - URL 處理
- Python - 泛型
- Python 庫
- NumPy 教程
- Pandas 教程
- SciPy 教程
- Matplotlib 教程
- Django 教程
- OpenCV 教程
- Python 雜項
- Python - 日期 & 時間
- Python - 數學
- Python - 迭代器
- Python - 生成器
- Python - 閉包
- Python - 裝飾器
- Python - 遞迴
- Python - 正則表示式
- Python - PIP
- Python - 資料庫訪問
- Python - 弱引用
- Python - 序列化
- Python - 模板
- Python - 輸出格式化
- Python - 效能測量
- Python - 資料壓縮
- Python - CGI 程式設計
- Python - XML 處理
- Python - GUI 程式設計
- Python - 命令列引數
- Python - 文件字串
- Python - JSON
- Python - 傳送郵件
- Python - 擴充套件
- Python - 工具/實用程式
- Python - GUIs
- Python 高階概念
- Python - 抽象基類
- Python - 自定義異常
- Python - 高階函式
- Python - 物件內部
- Python - 記憶體管理
- Python - 元類
- Python - 使用元類進行超程式設計
- Python - 模擬和存根
- Python - 猴子補丁
- Python - 訊號處理
- Python - 型別提示
- Python - 自動化教程
- Python - Humanize 包
- Python - 上下文管理器
- Python - 協程
- Python - 描述符
- Python - 診斷和修復記憶體洩漏
- Python - 不可變資料結構
- Python 有用資源
- Python - 問答
- Python - 線上測驗
- Python - 快速指南
- Python - 參考
- Python - 速查表
- Python - 專案
- Python - 有用資源
- Python - 討論
- Python 編譯器
- NumPy 編譯器
- Matplotlib 編譯器
- SciPy 編譯器
Python - 執行緒死鎖
死鎖可以描述為一種併發故障模式。它是在程式中出現的一種情況,其中一個或多個執行緒等待一個永遠不會發生的條件。結果,執行緒無法繼續執行,程式卡住或凍結,並且必須手動終止。
死鎖情況可能在您的併發程式中以多種方式出現。死鎖永遠不會有意開發,相反,它們實際上是程式碼中的副作用或錯誤。
執行緒死鎖的常見原因列在下面:
嘗試兩次獲取同一互斥鎖的執行緒。
相互等待的執行緒(例如 A 等待 B,B 等待 A)。
當執行緒未能釋放資源(例如鎖、訊號量、條件、事件等)。
以不同順序獲取互斥鎖的執行緒(例如,未能執行鎖排序)。
如何在 Python 執行緒中避免死鎖
當多執行緒應用程式中的多個執行緒嘗試訪問同一資源時,例如對同一檔案執行讀/寫操作,可能會導致資料不一致。因此,使用鎖定機制同步對資源的併發訪問非常重要。
Python 的threading模組提供了一種易於實現的鎖定機制來同步執行緒。您可以透過呼叫Lock()類建立一個新的鎖物件,該類將鎖初始化為未鎖定狀態。
使用 Lock 物件的鎖定機制
Lock類的物件有兩種可能的狀態:鎖定或未鎖定,最初在建立時處於未鎖定狀態。鎖不屬於任何特定執行緒。
Lock 類定義了acquire() 和 release() 方法。
acquire() 方法
acquire()方法更改鎖的狀態從未鎖定到鎖定。除非可選的 blocking 引數設定為 True,否則它會立即返回,在這種情況下,它會等待直到獲取鎖。
以下是此方法的語法:
Lock.acquire(blocking, timeout)
其中,
blocking - 如果設定為 False,則表示不阻塞。如果帶有 blocking 設定為 True 的呼叫將被阻塞,則立即返回 False;否則,將鎖設定為鎖定並返回 True。
timeout - 指定獲取鎖的超時時間。
此方法的返回值為 True,如果鎖成功獲取;否則為 False。
release() 方法
當狀態被鎖定時,另一個執行緒中的此方法將其更改為解鎖狀態。這可以從任何執行緒呼叫,而不僅僅是從獲取鎖的執行緒呼叫。
以下是release()方法的語法:
Lock.release()
release()方法只能在鎖定狀態下呼叫。如果嘗試釋放一個未鎖定的鎖,將會引發RuntimeError錯誤。
當鎖被鎖定時,將其重置為解鎖狀態,並返回。如果任何其他執行緒被阻塞等待鎖解鎖,則允許其中恰好一個執行緒繼續執行。此方法沒有返回值。
示例
在下面的程式中,兩個執行緒嘗試呼叫synchronized()方法。其中一個執行緒獲取鎖並獲得訪問許可權,而另一個執行緒則等待。當第一個執行緒的run()方法完成時,鎖被釋放,第二個執行緒可以使用synchronized方法。
當兩個執行緒都join後,程式結束。
from threading import Thread, Lock import time lock=Lock() threads=[] class myThread(Thread): def __init__(self,name): Thread.__init__(self) self.name=name def run(self): lock.acquire() synchronized(self.name) lock.release() def synchronized(threadName): print ("{} has acquired lock and is running synchronized method".format(threadName)) counter=5 while counter: print ('**', end='') time.sleep(2) counter=counter-1 print('\nlock released for', threadName) t1=myThread('Thread1') t2=myThread('Thread2') t1.start() threads.append(t1) t2.start() threads.append(t2) for t in threads: t.join() print ("end of main thread")
它將產生以下輸出:
Thread1 has acquired lock and is running synchronized method ********** lock released for Thread1 Thread2 has acquired lock and is running synchronized method ********** lock released for Thread2 end of main thread
用於同步的訊號量物件
除了鎖之外,Python的threading模組還支援訊號量,它提供了一種其他的同步技術。它是著名的計算機科學家Edsger W. Dijkstra發明的一種最古老的同步技術之一。
訊號量的基本概念是使用一個內部計數器,每個acquire()呼叫都會遞減它,每個release()呼叫都會遞增它。計數器永遠不會低於零;當acquire()發現它為零時,它會阻塞,等待直到其他執行緒呼叫release()。
threading模組中的Semaphore類定義了acquire()和release()方法。
acquire() 方法
如果在進入時內部計數器大於零,則將其減一併立即返回True。
如果在進入時內部計數器為零,則阻塞直到被對release()的呼叫喚醒。一旦被喚醒(並且計數器大於0),則將其減一併返回True。每個對release()的呼叫將喚醒恰好一個執行緒。執行緒喚醒的順序是任意的。
如果blocking引數設定為False,則不阻塞。如果一個沒有引數的呼叫會阻塞,則立即返回False;否則,執行與沒有引數呼叫時相同的事情,並返回True。
release() 方法
釋放一個訊號量,將內部計數器加一。當它在進入時為零,並且其他執行緒正在等待它再次大於零時,喚醒n個這些執行緒。
示例
此示例演示瞭如何在Python中使用Semaphore物件來控制多個執行緒對共享資源的訪問,以避免Python多執行緒程式中的死鎖。
from threading import * import time # creating thread instance where count = 3 lock = Semaphore(4) # creating instance def synchronized(name): # calling acquire method lock.acquire() for n in range(3): print('Hello! ', end = '') time.sleep(1) print( name) # calling release method lock.release() # creating multiple thread thread_1 = Thread(target = synchronized , args = ('Thread 1',)) thread_2 = Thread(target = synchronized , args = ('Thread 2',)) thread_3 = Thread(target = synchronized , args = ('Thread 3',)) # calling the threads thread_1.start() thread_2.start() thread_3.start()
它將產生以下輸出:
Hello! Hello! Hello! Thread 1 Hello! Thread 2 Thread 3 Hello! Hello! Thread 1 Hello! Thread 3 Thread 2 Hello! Hello! Thread 1 Thread 3 Thread 2