GIL全局解釋器鎖、死鎖、遞歸鎖、線程隊列

来源:https://www.cnblogs.com/setcreed/archive/2019/10/23/11729265.html
-Advertisement-
Play Games

[TOC] GIL全局解釋鎖 1. GIL本質上是一個互斥鎖。 2. GIL是為了阻止同一個進程內多個進程同時執行(並行) 單個進程下的多個線程無法實現並行,但能實現併發 3. 這把鎖主要是因為Cpython的記憶體管理不是線程安全的 保證線程在執行任務時不會被垃圾回收機制回收 多線程的作用 1. 計 ...


目錄

GIL全局解釋鎖

  1. GIL本質上是一個互斥鎖。
  2. GIL是為了阻止同一個進程內多個進程同時執行(並行)
    • 單個進程下的多個線程無法實現並行,但能實現併發
  3. 這把鎖主要是因為Cpython的記憶體管理不是線程安全的

    • 保證線程在執行任務時不會被垃圾回收機制回收
from threading import Thread
import time

num = 100


def task():
    global num
    num2 = num
    time.sleep(1)
    num = num2 - 1
    print(num)


for line in range(100):
    t = Thread(target=task)
    t.start()
    
# 這裡的運行結果都是99, 加了IO操作,所有線程都對num進行了減值操作,由於GIL鎖的存在,沒有修改成功,都是99

多線程的作用

  1. 計算密集型, 有四個任務,每個任務需要10s

單核:

  • 開啟進程
    • 消耗資源過大
    • 4個進程: 40s
  • 開啟線程
    • 消耗資源遠小於進程
    • 4個線程: 40s

多核:

  • 開啟進程
    • 並行執行, 效率比較高
    • 4個進程: 10s
  • 開啟線程
    • 併發執行,執行效率低
    • 4個線程: 40s
  1. IO密集型, 四個任務, 每個任務需要10s

單核:

  • 開啟進程
    • 消耗資源過大
    • 4個進程: 40s
  • 開啟線程
    • 消耗資源遠小於進程
    • 4個線程: 40s

多核:

  • 開啟進程
    • 並行執行, 效率小於多線程, 但是遇到IO會立馬切換CPU的執行許可權
    • 4個進程: 40s + 開啟進程消耗的額外時間
  • 開啟線程
    • 併發執行,執行效率高於多進程
    • 4個線程: 40s

測試計算密集型

from threading import Thread
from multiprocessing import Process
import time
import os


# 計算密集型
def work1():
    number = 0
    for line in range(100000000):
        number += 1

# IO密集型
def work2():
    time.sleep(2)


if __name__ == '__main__':
    # 測試計算密集型
    print(os.cpu_count())   # 4核cpu

    start = time.time()
    list1 = []
    for line in range(6):

        p = Process(target=work1)  # 程式執行時間8.756593704223633
        # p = Thread(target=work1)   # 程式執行時間31.78555393218994
        list1.append(p)
        p.start()
    for p in list1:
        p.join()
    end = time.time()
    print(f'程式執行時間{end - start}')

IO密集型

from threading import Thread
from multiprocessing import Process
import time
import os

# 計算密集型
def work1():
    number = 0
    for line in range(100000000):
        number += 1

# IO密集型
def work2():
    time.sleep(1)


if __name__ == '__main__':
    # 測試計算密集型
    print(os.cpu_count())   # 4核cpu

    start = time.time()
    list1 = []
    for line in range(100):

        # p = Process(target=work2)  # 程式執行時間15.354223251342773
        p = Thread(target=work2)   # 程式執行時間1.0206732749938965
        list1.append(p)
        p.start()
    for p in list1:
        p.join()
    end = time.time()
    print(f'程式執行時間{end - start}')

結論:

  • 在計算密集型的情況下, 使用多進程
  • 在IO密集型的情況下, 使用多線程
  • 高效執行多個進程, 內有多個IO密集型程式,使用多進程 + 多線程

死鎖現象

指兩個或兩個以上的進程或線程在執行過程中,因爭奪資源而造成的一種互相等待的現象,如無外力作用,它們都無法推進下去.此時稱系統處於死鎖狀態

以下就是死鎖:

from threading import Thread, Lock
from threading import current_thread
import time

mutex_a = Lock()
mutex_b = Lock()


class MyThread(Thread):

    def run(self):
        self.func1()
        self.func2()

    def func1(self):
        mutex_a.acquire()
        print(f'用戶{self.name}搶到鎖a')
        mutex_b.acquire()
        print(f'用戶{self.name}搶到鎖b')
        mutex_b.release()
        print(f'用戶{self.name}釋放鎖b')
        mutex_a.release()
        print(f'用戶{self.name}釋放鎖a')

    def func2(self):
        mutex_b.acquire()
        print(f'用戶{self.name}搶到鎖b')
        time.sleep(1)
        mutex_a.acquire()
        print(f'用戶{self.name}搶到鎖a')
        mutex_a.release()
        print(f'用戶{self.name}釋放鎖a')
        mutex_b.release()
        print(f'用戶{self.name}釋放鎖b')


for line in range(10):
    t = MyThread()
    t.start()
    
    
'''
用戶Thread-1搶到鎖a
用戶Thread-1搶到鎖b
用戶Thread-1釋放鎖b
用戶Thread-1釋放鎖a
用戶Thread-1搶到鎖b
用戶Thread-2搶到鎖a
'''
# 一直等待

遞歸鎖

用於解決死鎖問題

RLock: 比喻成萬能鑰匙,可以提供給多個人使用

但是第一個使用的時候,會對該鎖做一個引用計數

只有引用計數為0, 才能真正釋放讓一個人使用

上面的例子中用RLock代替Lock, 就不會發生死鎖現象

from threading import Thread, Lock, RLock
from threading import current_thread
import time

# mutex_a = Lock()
# mutex_b = Lock()

mutex_a = mutex_b = RLock()

class MyThread(Thread):

    def run(self):
        self.func1()
        self.func2()

    def func1(self):
        mutex_a.acquire()
        print(f'用戶{self.name}搶到鎖a')
        mutex_b.acquire()
        print(f'用戶{self.name}搶到鎖b')
        mutex_b.release()
        print(f'用戶{self.name}釋放鎖b')
        mutex_a.release()
        print(f'用戶{self.name}釋放鎖a')

    def func2(self):
        mutex_b.acquire()
        print(f'用戶{self.name}搶到鎖b')
        time.sleep(1)
        mutex_a.acquire()
        print(f'用戶{self.name}搶到鎖a')
        mutex_a.release()
        print(f'用戶{self.name}釋放鎖a')
        mutex_b.release()
        print(f'用戶{self.name}釋放鎖b')


for line in range(10):
    t = MyThread()
    t.start()

信號量(瞭解)

互斥鎖: 比喻成一個家用馬桶, 同一時間只能讓一個人去使用

信號比喻成公測多個馬桶: 同一時間可以讓多個人去使用

from threading import Semaphore
from threading import Thread
from threading import current_thread
import time

sm = Semaphore(5)

def task():
    sm.acquire()
    print(f'{current_thread().name}執行任務')
    time.sleep(1)
    sm.release()


for i in range(20):
    t = Thread(target=task)
    t.start()

線程隊列

線程Q: 就是線程隊列 FIFO

  • 普通隊列: 先進先出 FIFO
  • 特殊隊列: 後進先出 LIFO
  • 優先順序隊列: 若傳入一個元組,會依次判斷參數的ASCII的數值大小
import queue

# 普通的線程隊列: 遵循先進先出
q = queue.Queue()
q.put(1)
q.put(2)
q.put(3)

print(q.get())  # 1
print(q.get())  # 2


# LIFO隊列  後進先出
q = queue.LifoQueue()
q.put(1)
q.put(2)
q.put(3)
print(q.get())  # 3


# 優先順序隊列:根據參數內
q = queue.PriorityQueue()
q.put((4, '我'))
q.put((2, '你'))
q.put((3, 'ta'))
print(q.get())   # (2, '你')

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 原文地址: "梁桂釗的博客" 博客地址: "http://blog.720ui.com" 歡迎關註公眾號:「服務端思維」。一群同頻者,一起成長,一起精進,打破認知的局限性。 今天,探討一個有趣的話題:我們可以通過 Git 來實現項目版本控制;通過 Jenkins 進行持續集成,那麼對於資料庫層面,我 ...
  • 集合類 Collection介面 定義的是所有單列集合中共性方法 創建對象使用多態 Collection<String> coll = new ArrayList<>() add() 把給定的對象添加到當前集合中,返回一個boolean值 remove() 在集合中刪除指定的對象,返回一個boole ...
  • 一、json數據 python數據類型與json對應關係表如下: 以上就是關於json的所有數據類型。 什麼是json? 定義: 講json對象,不得不提到JS對象: 合格的json對象: 不合格的json對象: python數據類型和json的轉換 JavaScript數據類型和json的轉換 p ...
  • 前面的章節,講解了[Spring Boot集成Spring Cache]( https://blog.csdn.net/Simple_Yangger/article/details/102693316),Spring Cache已經完成了多種Cache的實現,包括EhCache、RedisCache ...
  • 前面的章節,講解了[Spring Boot集成Spring Cache]( https://blog.csdn.net/Simple_Yangger/article/details/102693316),Spring Cache已經完成了多種Cache的實現,包括EhCache、RedisCache ...
  • 在python腳本中我們經常看到如下的代碼: # hello.py def hello(): print("hello world!") def test(): hello() if __name__ == '__main__': test() 通常,一個python文件有兩種使用方法: (1)直接 ...
  • 繼承Thread類 實現Runnable介面 匿名內部類的兩種寫法 基於java.util.concurrent.Callable工具類的實現,帶返回值 基於java.util.Timer工具類的實現 基於java.util.concurrent.Executors工具類,基於線程池的實現 更多信息 ...
  • 集成 websocket 的四種方案 1. 原生註解 pom.xml WebSocketConfig 說明: 這個配置類很簡單,通過這個配置 spring boot 才能去掃描後面的關於 websocket 的註解 WsServerEndpoint 說明 這裡有幾個註解需要註意一下,首先是他們的包都 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...