詳解Python多線程下的list
list 是 Python 常用的幾個(gè)基本數(shù)據(jù)類型之一.正常情況下我們會(huì)對(duì) list 有增刪改查的操作,顯然易見不會(huì)有任何問題.那么如果我們?cè)囍诙嗑€程下操作list 會(huì)有問題嗎?
多線程下的 list
安全 or 不安全? 不安全!
通常我們說(shuō)的線程安全是指針對(duì)某個(gè)數(shù)據(jù)結(jié)構(gòu)的所有操作都是線程安全,在這種定義下,Python 常用的數(shù)據(jù)結(jié)構(gòu) list,dict,str 等都是線程不安全的
盡管多線程下的 list 是線程不安全的,但是在 append 的操作下是它又是線程安全的.
如何判斷線程安全呢?
對(duì)于線程安全不安全,我們可以通過(guò)極端條件下去復(fù)現(xiàn),從而得出結(jié)論。比如說(shuō)判斷 list 是否線程安全
import threadingimport time# 隨意設(shè)置 count 的值,值越大錯(cuò)誤拋出的越快count = 1000l = []def add(): for i in range(count): l.append(i) time.sleep(0.0001)def remove(): for i in range(count): l.remove(i) time.sleep(0.0001)t1 = threading.Thread(target=add)t2 = threading.Thread(target=remove)t1.start()t2.start()t1.join()t2.join()print(l)
有時(shí)候一次運(yùn)行并不一定就會(huì)出錯(cuò),多次重試之后會(huì)出現(xiàn)類似下面的錯(cuò)誤
很顯然這種操作方式不具有普適性,如果要是歐氣太強(qiáng),說(shuō)不定會(huì)一直不出現(xiàn)異常。
那么出了這種方式,有沒有比較簡(jiǎn)單有效的方法嗎?答案是有的
dis
dis 庫(kù)是 Python 自帶的一個(gè)庫(kù),可以用來(lái)分析字節(jié)碼。這里我們需要有這樣的認(rèn)識(shí),字節(jié)碼的每一行都是一個(gè)原子操作,多線程切換就是以原子操作為單位的,如果一個(gè)操作需要兩行字節(jié)碼就說(shuō)明它是線程不安全的
remove
這里我們先看一下上面 list 的 remove 操作
>>> import dis>>> def test_remove():... a = [1]... a.remove(0)... >>> dis.dis(test_remove) 2 0 LOAD_CONST1 (1) 2 BUILD_LIST1 4 STORE_FAST0 (a) 3 6 LOAD_FAST0 (a) 8 LOAD_ATTR0 (remove) 10 LOAD_CONST2 (0) 12 CALL_FUNCTION 1 14 POP_TOP 16 LOAD_CONST0 (None) 18 RETURN_VALUE
從上面不難看出,整個(gè) remove 操作被分成了好幾條指令,這就意味著在多線程情況下會(huì)出現(xiàn)錯(cuò)亂的情況,試想一下,如果多線程下都去 remove 列表的話,并且不按照順序,很容易出現(xiàn)問題。
append
在最上面我們說(shuō)到,list 的 append 操作是線程安全的,那么究竟是為什么呢?我們同樣來(lái)用 dis 查看一下
8 19 LOAD_GLOBAL 0 (a) 22 LOAD_ATTR2 (append) 25 LOAD_CONST2 (1) 28 CALL_FUNCTION 1 31 POP_TOP
這里顯然,append 也是有幾條指令,勢(shì)必在多線程執(zhí)行的情況下也會(huì)發(fā)生交錯(cuò),但是對(duì)于多線程下我們操作 append, 我們肯定也不會(huì)在乎這個(gè)時(shí)候 list 到順序問題了,所以我們說(shuō)它的 append 是線程安全的
參考
https://stackoverflow.com/questions/6319207/are-lists-thread-safe/19728536#19728536
https://docs.python.org/3/faq/library.html#what-kinds-of-global-value-mutation-are-thread-safe
以上就是詳解Python多線程下的list的詳細(xì)內(nèi)容,更多關(guān)于Python多線程下的list的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. ASP.NET MVC把數(shù)據(jù)庫(kù)中枚舉項(xiàng)的數(shù)字轉(zhuǎn)換成文字2. SpringMVC+Jquery實(shí)現(xiàn)Ajax功能3. 基于javaweb+jsp實(shí)現(xiàn)企業(yè)財(cái)務(wù)記賬管理系統(tǒng)4. 博客日志摘要暨RSS技術(shù)5. 低版本IE正常運(yùn)行HTML5+CSS3網(wǎng)站的3種解決方案6. WML學(xué)習(xí)之一概述和基本規(guī)則7. Java pom.xml parent引用報(bào)錯(cuò)問題解決方案8. python 集合set中 add與update區(qū)別介紹9. CSS單標(biāo)簽實(shí)現(xiàn)復(fù)雜的棋盤布局10. Java try catch finally異常處理組合詳解
