校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃

主頁 > 知識庫 > Python threading Local()函數用法案例詳解

Python threading Local()函數用法案例詳解

熱門標簽:鶴壁手機自動外呼系統違法嗎 高德地圖標注收入咋樣 萊蕪電信外呼系統 B52系統電梯外呼顯示E7 沈陽防封電銷電話卡 怎么辦理400客服電話 企業微信地圖標注 地圖標注多個 銀川電話機器人電話

前言

當多線程訪問同一個公共資源時,如果涉及到修改該公共資源的操作就可能會出現由于數據不同步導致的線程安全問題。一般情況下我們可以通過給公共資源加互斥鎖的方式來處理該問題。

當然,除非必須將多線程使用的資源設置為公共資源的情況。如果一個資源不需要在多個線程之間共享。我們也可以使用Python threading模塊提供的local()方式來避免線程安全問題。
Python threading模塊的local()函數跟Java中的ThreadLocal類有諸多類似的地方,感興趣的小伙伴可以看下Java版的ThreadLoalJava ThreadLocal原理解析以及應用場景分析案例詳解

local() 函數是什么?

threading的local()函數主要是用來封裝公共資源,使得同一個公共資源在不同線程之間得以隔離。這句話該如何理解呢?舉個例子說明下!假設現在有一個大箱子(相當于公共資源),每個人(相當于各個線程)將自己的手機放入這個大箱子里。如果不做任何控制的話,當人們從大箱子中取出手機時極有可能會出現取錯的情況(找不到自己當初放入的手機)。而使用local()函數的話,就相當于對這個大箱子進行管理。當每個人放入手機的時候做一個標記(比如在手機上標記所有者的姓名)并隔離放置到箱子中。這樣當人們從大箱子中取出手機就能準確的找到自己當初放入的手機。

調用local()函數會生成一個ThreadLocal對象,該對象是所有線程都能訪問的,就像上面例子中的大箱子。但是,放入到ThreadLocal對象中的變量則是各個線程所獨有的,隨便變量名相同,但是指向的值則是完全不同的。

local()函數如何用?

local()函數使用的基本語法是:

import threading

local=threading.local()

第一步就是引入threading模塊,第二步就是調用local()函數得到全局的Threadlocal對象。這樣說始終是有點干澀,沒味道。那么就給代碼加點鹽吧。還是從那個大箱子說起。

1. 不做標記,不做隔離

第一個示例代碼就是所有人將自己的手機放入大箱子里,不做標記,不做隔離。先放入,過一段時間后再取出。

import threading
import time


def set_telephone(telephone):
    global global_telephone
    global_telephone = telephone
    print(threading.current_thread().name + " 放入的手機是", global_telephone)
    time.sleep(1)
    get_telephone()


def get_telephone():
    print(threading.current_thread().name + " 取出的手機是", global_telephone)


if __name__ == '__main__':
    for i in range(3):
        thread = threading.Thread(target=set_telephone, name='學生' + str(i), args=('手機' + str(i),))
        thread.start()

運行結果是:

學生0 放入的手機是 手機0

學生1 放入的手機是 手機1

學生2 放入的手機是 手機2

學生0 取出的手機是 手機2

學生1 取出的手機是 手機2

學生2 取出的手機是 手機2

這里有三個線程,分別模擬學生0,學生1,學生2 將各種的手機賦值給一個全局變量global_telephone(大箱子),然后取全局變量global_telephone中的值。可以看出取出的結果都變成了手機2。這顯然沒有達到我們的預期結果。這就是不加控制的后果。

2.使用local()函數加以控制

使用local()函數控制的話,就是將全局變量替換成ThreadLoal對象,由他來管理每個線程中的值。

import threading
import time


def set_telephone(telephone):
    local.telephone = telephone
    print(threading.current_thread().name + " 放入的手機是", local.telephone + "\n")
    time.sleep(1)
    get_telephone()


def get_telephone():
    print(threading.current_thread().name + " 取出的手機是", local.telephone + "\n")


if __name__ == '__main__':
    local = threading.local()
    for i in range(3):
        thread = threading.Thread(target=set_telephone, name='學生' + str(i), args=('手機' + str(i),))
        thread.start()

運行結果是:

學生0 放入的手機是 手機0

學生1 放入的手機是 手機1

學生2 放入的手機是 手機2

學生1 取出的手機是 手機1

學生0 取出的手機是 手機0

學生2 取出的手機是 手機2

可以看出每個學生放入的手機和最終取出的手機是一致的。那么threading的local()函數是如何實現這一效果的呢?我們在這里不妨做一個推理。應該是將手機和它的主人做了一層映射關系。根據主人的唯一標識來尋找自己的手機。

3. 模擬實現local()的功能,創建一個箱子

前面我們推測我們需要定義一個全局的字典來存放每個學生各自放入的手機,字典的鍵是線程ID,值是指定的鍵值對。示例代碼如下:

import threading
import time

global_goods_dict = {}

# {
#     "線程ID":{"telephone":"放入的具體手機"},
#     "線程ID":{"telephone":"放入的具體手機"},
#     "線程ID":{"telephone":"放入的具體手機"}
#
# }

def set_telephone(telephone):
    # 獲取線程ID
    thread_id = threading.get_ident()
    global_goods_dict[thread_id] = {}
    global_goods_dict[thread_id]["telephone"] = telephone
    print(threading.current_thread().name + " 放入的手機是", telephone)
    time.sleep(1)
    get_telephone()


def get_telephone():
    thread_id = threading.get_ident()
    print(threading.current_thread().name + " 取出的手機是", global_goods_dict[thread_id]["telephone"])


if __name__ == '__main__':
    for i in range(3):
        thread = threading.Thread(target=set_telephone, name='學生' + str(i), args=('手機' + str(i),))
        thread.start()

運行結果同上,這里定義了一個全局的字典global_goods_dict,字典的鍵盤是線程ID,這就保證了每個線程只能取到自己設置的數據。字典的值同樣是一個字典。這是因為一個線程的要存的值可能不止一個。這里的global_goods_dict[thread_id]["telephone"] = telephone 就等價于上例中的local.telephone = telephone。這樣使用雖然能達到效果,但是使用起來還是有點繁瑣。那么能不能想local()函數那樣使用起來絲滑呢。

4. 簡化代碼操作,進一步模擬實現local()函數

我們可以將全局的global_goods_dict字典用一個類封裝到一個類中。讓該類在自動的設置值

class MyBox:
    box = {}

    def __setattr__(self, key, value):
        thread_id = threading.get_ident()
        # 單元格已存在
        if thread_id in MyBox.box:
            MyBox.box[thread_id][key] = value
        else:
            MyBox.box[thread_id] = {key: value}

    def __getattr__(self, item):
        thread_id = threading.get_ident()
        return MyBox.box[thread_id][item]


def set_telephone(telephone):
    myBox.telephone = telephone
    print(threading.current_thread().name + " 放入的手機是", myBox.telephone + "\n")
    time.sleep(1)
    get_telephone()


def get_telephone():
    print(threading.current_thread().name + " 取出的手機是", myBox.telephone + "\n")


if __name__ == '__main__':
    myBox = MyBox()
    for i in range(3):
        thread = threading.Thread(target=set_telephone, name='學生' + str(i), args=('手機' + str(i),))
        thread.start()

運行結果同上。這里通過MyBox類封裝了一個名為box的字典。該字典的鍵是當前線程ID,值是賦值的變量名以及值組成的鍵值對。當執行set_telephone方法的myBox.telephone = telephone
,實際上會調用MyBox的__setattr__方法,參數key是telephone,參數value是"手機xx"。當調用myBox.telephone時實際上會調用__getattr__方法,傳入的參數item是telephone。取值時首先獲取當前線程ID。

總結

本文從實際例子出發詳細介紹了threading模塊的local()函數的使用。

到此這篇關于Python threading Local()函數用法案例詳解的文章就介紹到這了,更多相關Python threading Local()函數內容請搜索腳本之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • Python threading.local代碼實例及原理解析
  • python語言線程標準庫threading.local解讀總結
  • python threading模塊的使用指南
  • Python中threading庫實現線程鎖與釋放鎖
  • python中threading和queue庫實現多線程編程

標簽:葫蘆島 湘西 三亞 呼倫貝爾 銀川 呼倫貝爾 安慶 烏魯木齊

巨人網絡通訊聲明:本文標題《Python threading Local()函數用法案例詳解》,本文關鍵詞  Python,threading,Local,函數,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Python threading Local()函數用法案例詳解》相關的同類信息!
  • 本頁收集關于Python threading Local()函數用法案例詳解的相關信息資訊供網民參考!
  • 推薦文章
    主站蜘蛛池模板: 珲春市| 彩票| 富阳市| 呼图壁县| 三明市| 通山县| 兴隆县| 武夷山市| 镇远县| 教育| 特克斯县| 古蔺县| 许昌县| 灵丘县| 乌拉特后旗| 合川市| 红河县| 新宁县| 岑巩县| 哈尔滨市| 新疆| 江达县| 西青区| 临沧市| 页游| 鸡西市| 绩溪县| 长汀县| 山阳县| 客服| 徐闻县| 阿勒泰市| 贵州省| 白水县| 益阳市| 古交市| 邵武市| 富民县| 宝山区| 宿松县| 曲麻莱县|