溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶(hù)服務(wù)條款》

詳解python多線程、鎖、event事件機(jī)制的簡(jiǎn)單使用

發(fā)布時(shí)間:2020-09-25 02:04:12 來(lái)源:腳本之家 閱讀:237 作者:君惜 欄目:開(kāi)發(fā)技術(shù)

線程和進(jìn)程

1、線程共享創(chuàng)建它的進(jìn)程的地址空間,進(jìn)程有自己的地址空間

2、線程可以訪問(wèn)進(jìn)程所有的數(shù)據(jù),線程可以相互訪問(wèn)

3、線程之間的數(shù)據(jù)是獨(dú)立的

4、子進(jìn)程復(fù)制線程的數(shù)據(jù)

5、子進(jìn)程啟動(dòng)后是獨(dú)立的 ,父進(jìn)程只能殺掉子進(jìn)程,而不能進(jìn)行數(shù)據(jù)交換

6、修改線程中的數(shù)據(jù),都是會(huì)影響其他的線程,而對(duì)于進(jìn)程的更改,不會(huì)影響子進(jìn)程

threading.Thread

Thread 是threading模塊中最重要的類(lèi)之一,可以使用它來(lái)創(chuàng)建線程。有兩種方式來(lái)創(chuàng)建線程:一種是通過(guò)繼承Thread類(lèi),重寫(xiě)它的run方法;另一種是創(chuàng)建一個(gè)threading.Thread對(duì)象,在它的初始化函數(shù)(__init__)中將可調(diào)用對(duì)象作為參數(shù)傳入。
先來(lái)看看通過(guò)繼承threading.Thread類(lèi)來(lái)創(chuàng)建線程的例子:

import threading
import time

class MyThread(threading.Thread):
 def __init__(self, arg):
  # super(MyThread, self).__init__() # 新式類(lèi)繼承原有方法寫(xiě)法
  threading.Thread.__init__(self)
  self.arg = arg

 def run(self):
  time.sleep(2)
  print(self.arg)

for i in range(10):
 thread = MyThread(i)
 print(thread.name)
 thread.start()

另外一種創(chuàng)建線程的方法:

import threading
import time

def process(arg):
 time.sleep(2)
 print(arg)

for i in range(10):
 t = threading.Thread(target=process, args=(i,))
 print(t.name)
 t.start()

Thread類(lèi)還定義了以下常用方法與屬性:

Thread.getName() 獲取線程名稱(chēng)

Thread.setName() 設(shè)置線程名稱(chēng)

Thread.name 線程名稱(chēng)

Thread.ident 獲取線程的標(biāo)識(shí)符。線程標(biāo)識(shí)符是一個(gè)非零整數(shù),只有在調(diào)用了start()方法之后該屬性才有效,否則它只返回None

判斷線程是否是激活的(alive)。從調(diào)用start()方法啟動(dòng)線程,到run()方法執(zhí)行完畢或遇到未處理異常而中斷 這段時(shí)間內(nèi),線程是激活的

Thread.is_alive()
Thread.isAlive()

Thread.join([timeout]) 調(diào)用Thread.join將會(huì)使主調(diào)線程堵塞,直到被調(diào)用線程運(yùn)行結(jié)束或超時(shí)。參數(shù)timeout是一個(gè)數(shù)值類(lèi)型,表示超時(shí)時(shí)間,如果未提供該參數(shù),那么主調(diào)線程將一直堵塞到被調(diào)線程結(jié)束

Python GIL(Global Interpreter Lock)

GIL并不是Python的特性,它是在實(shí)現(xiàn)Python解析器(CPython)時(shí)所引入的一個(gè)概念。就好比C++是一套語(yǔ)言(語(yǔ)法)標(biāo)準(zhǔn),但是可以用不同的編譯器來(lái)編譯成可執(zhí)行代碼。有名的編譯器例如GCC,INTEL C++,Visual C++等。Python也一樣,同樣一段代碼可以通過(guò)CPython,PyPy,Psyco等不同的Python執(zhí)行環(huán)境來(lái)執(zhí)行。像其中的JPython就沒(méi)有GIL。然而因?yàn)镃Python是大部分環(huán)境下默認(rèn)的Python執(zhí)行環(huán)境。所以在很多人的概念里CPython就是Python,也就想當(dāng)然的把GIL歸結(jié)為Python語(yǔ)言的缺陷。所以這里要先明確一點(diǎn):GIL并不是Python的特性,Python完全可以不依賴(lài)于GIL。

線程鎖的使用:

# 鎖:GIL 全局解釋器 它是為了保證線程在運(yùn)行過(guò)程中不被搶占
number = 0
lock = threading.RLock() # 創(chuàng)建鎖


def run(num):
 lock.acquire() # 加鎖
 global number
 number += 1
 print(number)
 time.sleep(2)
 lock.release() # 釋放鎖

for i in range(10):
 t = threading.Thread(target=run, args=(i, ))
 t.start()

Join & Daemon

主線程A中,創(chuàng)建了子線程B,并且在主線程A中調(diào)用了B.setDaemon(),這個(gè)的意思是,把主線程A設(shè)置為守護(hù)線程,這時(shí)候,要是主線程A執(zhí)行結(jié)束了,就不管子線程B是否完成,一并和主線程A退出.這就是setDaemon方法的含義,這基本和join是相反的。此外,還有個(gè)要特別注意的:必須在start() 方法調(diào)用之前設(shè)置,如果不設(shè)置為守護(hù)線程,程序會(huì)被無(wú)限掛起。

class MyThread1(threading.Thread):
 def __init__(self):
  threading.Thread.__init__(self)

 def run(self):
  print("thread start")
  time.sleep(3)
  print('thread end')

print('main start')
thread1 = MyThread1()
# thread1.setDaemon(True)  # 設(shè)置子線程是否跟隨主線程一起結(jié)束
thread1.start()
time.sleep(1)
print('satrt join')
# thread1.join() # 使主線程阻塞,直至子線程運(yùn)行完畢再繼續(xù)主線程
print('end join')
def run(n):
 print('[%s]------running----\n' % n)
 time.sleep(2)
 print('--done--')


def main():
 for i in range(5):
  t = threading.Thread(target=run, args=[i,])
  t.start()
  # t.join()
  print('starting thread', t.getName())


m = threading.Thread(target=main,args=[])
# m.setDaemon(True) # 將主線程設(shè)置為Daemon線程,它退出時(shí),其它子線程會(huì)同時(shí)退出,不管是否執(zhí)行完任務(wù)
m.start()
# m.join() # 使主線程阻塞,直至子線程運(yùn)行完畢再繼續(xù)主線程
print("---main thread done----")

線程鎖(互斥鎖Mutex)

一個(gè)進(jìn)程下可以啟動(dòng)多個(gè)線程,多個(gè)線程共享父進(jìn)程的內(nèi)存空間,也就意味著每個(gè)線程可以訪問(wèn)同一份數(shù)據(jù),此時(shí),如果2個(gè)線程同時(shí)要修改同一份數(shù)據(jù),會(huì)出現(xiàn)什么狀況?

num = 100 # 設(shè)定一個(gè)共享變量
def subNum():
 global num # 在每個(gè)線程中都獲取這個(gè)全局變量
 print('--get num:', num)
 time.sleep(2)
 num -= 1 # 對(duì)此公共變量進(jìn)行-1操作
thread_list = []
for i in range(100):
 t = threading.Thread(target=subNum)
 t.start()
 thread_list.append(t)
for t in thread_list: # 等待所有線程執(zhí)行完畢
 t.join()
print('final num:', num)
# 加鎖版本
def subNum():
 global num # 在每個(gè)線程中都獲取這個(gè)全局變量
 print('--get num:', num)
 time.sleep(1)
 lock.acquire() # 修改數(shù)據(jù)前加鎖
 num -= 1 # 對(duì)此公共變量進(jìn)行-1操作
 lock.release() # 修改后釋放

num = 100 # 設(shè)定一個(gè)共享變量
thread_list = []
lock = threading.Lock() # 生成全局鎖
for i in range(100):
 t = threading.Thread(target=subNum)
 t.start()
 thread_list.append(t)
for t in thread_list: # 等待所有線程執(zhí)行完畢
 t.join()
print('final num:', num)

Rlock與Lock的區(qū)別:

RLock允許在同一線程中被多次acquire。而Lock卻不允許這種情況。否則會(huì)出現(xiàn)死循環(huán),程序不知道解哪一把鎖。注意:如果使用RLock,那么acquire和release必須成對(duì)出現(xiàn),即調(diào)用了n次acquire,必須調(diào)用n次的release才能真正釋放所占用的鎖

Events

Python提供了Event對(duì)象用于線程間通信,它是由線程設(shè)置的信號(hào)標(biāo)志,如果信號(hào)標(biāo)志位真,則其他線程等待直到信號(hào)接觸。
Event對(duì)象實(shí)現(xiàn)了簡(jiǎn)單的線程通信機(jī)制,它提供了設(shè)置信號(hào),清除信號(hào),等待等用于實(shí)現(xiàn)線程間的通信。

event = threading.Event() 創(chuàng)建一個(gè)event

1 設(shè)置信號(hào)
event.set()

使用Event的set()方法可以設(shè)置Event對(duì)象內(nèi)部的信號(hào)標(biāo)志為真。Event對(duì)象提供了isSet()方法來(lái)判斷其內(nèi)部信號(hào)標(biāo)志的狀態(tài)。
當(dāng)使用event對(duì)象的set()方法后,isSet()方法返回真

2 清除信號(hào)
event.clear()

使用Event對(duì)象的clear()方法可以清除Event對(duì)象內(nèi)部的信號(hào)標(biāo)志,即將其設(shè)為假,當(dāng)使用Event的clear方法后,isSet()方法返回假

3 等待
event.wait()

Event對(duì)象wait的方法只有在內(nèi)部信號(hào)為真的時(shí)候才會(huì)很快的執(zhí)行并完成返回。當(dāng)Event對(duì)象的內(nèi)部信號(hào)標(biāo)志位假時(shí),
則wait方法一直等待到其為真時(shí)才返回。也就是說(shuō)必須set新號(hào)標(biāo)志位真

def do(event):
 print('start')
 event.wait()
 print('execute')

event_obj = threading.Event()
for i in range(10):
 t = threading.Thread(target=do, args=(event_obj,))
 t.start()

event_obj.clear()
inp = input('輸入內(nèi)容:')
if inp == 'true':
 event_obj.set()

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持億速云。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI