溫馨提示×

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

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

python 線程條件變量Condition(31)

發(fā)布時(shí)間:2020-02-27 08:43:51 來源:網(wǎng)絡(luò) 閱讀:153 作者:qq5d6f345f0205e 欄目:編程語言

對(duì)于線程與線程之間的交互我們?cè)谇懊娴奈恼乱呀?jīng)介紹了 python 互斥鎖Lock / python事件Event , 今天繼續(xù)介紹一種線程交互方式 –?線程條件變量Condition.

?

python 線程條件變量Condition(31)

?

一.線程條件變量Condition相關(guān)函數(shù)介紹

acquire()?—? 線程鎖,注意線程條件變量Condition中的所有相關(guān)函數(shù)使用必須在acquire()?/release()?內(nèi)部操作;

release()?— 釋放鎖,注意線程條件變量Condition中的所有相關(guān)函數(shù)使用必須在acquire()?/release()?內(nèi)部操作;

wait(timeout)?—? 線程掛起(阻塞狀態(tài)),直到收到一個(gè)notify通知或者超時(shí)才會(huì)被喚醒繼續(xù)運(yùn)行(超時(shí)參數(shù)默認(rèn)不設(shè)置,可選填,類型是浮點(diǎn)數(shù),單位是秒)。wait()必須在已獲得Lock前提下才能調(diào)用,否則會(huì)觸發(fā)RuntimeError;

notify(n=1)?—? 通知其他線程,那些掛起的線程接到這個(gè)通知之后會(huì)開始運(yùn)行,缺省參數(shù),默認(rèn)是通知一個(gè)正等待通知的線程,最多則喚醒n個(gè)等待的線程。notify()必須在已獲得Lock前提下才能調(diào)用,否則會(huì)觸發(fā)RuntimeError,notify()不會(huì)主動(dòng)釋放Lock;

notifyAll()?—? 如果wait狀態(tài)線程比較多,notifyAll的作用就是通知所有線程;

?

python 線程條件變量Condition(31)

?

?

二.線程條件變量Condition原理

在前面的文章已經(jīng)介紹過互斥鎖,主要作用是并行訪問共享資源時(shí),保護(hù)共享資源,防止出現(xiàn)臟數(shù)據(jù)。python 條件變量Condition也需要關(guān)聯(lián)互斥鎖,同時(shí)Condition自身提供了wait/notify/notifyAll方法,用于阻塞/通知其他并行線程,可以訪問共享資源了??梢赃@么理解,Condition提供了一種多線程通信機(jī)制,假如線程1需要數(shù)據(jù),那么線程1就阻塞等待,這時(shí)線程2就去制造數(shù)據(jù),線程2制造好數(shù)據(jù)后,通知線程1可以去取數(shù)據(jù)了,然后線程1去獲取數(shù)據(jù)。

python 線程條件變量Condition(31)

三.線程條件變量Condition使用

案例一:成語接龍

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

85

86

87

88

89

# !usr/bin/env python

# -*- coding:utf-8 _*-

"""

@Author:何以解憂

@Blog(個(gè)人博客地址): shuopython.com

@WeChat Official Account(微信公眾號(hào)):猿說python

@Github:www.github.com

@File:python_.py

@Time:2019/10/21 21:25

?

@Motto:不積跬步無以至千里,不積小流無以成江海,程序人生的精彩需要堅(jiān)持不懈地積累!

"""

?

# 導(dǎo)入線程模塊

import threading

?

# 創(chuàng)建條件變量condition

con = threading.Condition()

?

def thread_one(name):

????# 條件變量condition 線程上鎖

????con.acquire()

?

????print("{}:成語接龍準(zhǔn)備好了嗎".format(name))

????# 喚醒正在等待(wait)的線程

????con.notify()

?

????# 等待對(duì)方回應(yīng)消息,使用wait阻塞線程,等待對(duì)方通過notify喚醒本線程

????con.wait()

????print("{}:一干二凈".format(name))

????# 喚醒對(duì)方

????con.notify()

?

????# 等待消息答應(yīng)

????con.wait()

????print("{}:一天就知道看抖音美女,給你來個(gè)簡(jiǎn)單點(diǎn)的,來了:毛手毛腳".format(name))

????# 喚醒對(duì)方

????con.notify()

?

????# 等待消息答應(yīng)

????con.wait()

????print("{}:喲喲喲,不錯(cuò)不錯(cuò)!".format(name))

????# 喚醒對(duì)方

????con.notify()

?

????# 條件變量condition 線程釋放鎖

????con.release()

?

def thread_two(name):

????# 條件變量condition 線程上鎖

????con.acquire()

?

????# wait阻塞狀態(tài),等待其他線程通過notify喚醒本線程

????con.wait()

????print("{}:準(zhǔn)備好了~開始吧!".format(name))

????# 喚醒對(duì)方

????con.notify()

?

????# 等待消息答應(yīng)

????con.wait()

????print("{}:凈你妹啊,沒法接...來個(gè)簡(jiǎn)單點(diǎn)的...".format(name))

????# 喚醒對(duì)方

????con.notify()

?

????# 等待消息答應(yīng)

????con.wait()

????print("{}:嘿,這個(gè)我知道:腳踏實(shí)地".format(name))

????# 喚醒對(duì)方

????con.notify()

?

????con.release()

?

if __name__ == "__main__":

?

????# 創(chuàng)建并初始化線程

????t1 = threading.Thread(target=thread_one,args=("A"))

????t2 = threading.Thread(target=thread_two,args=("B"))

?

????# 啟動(dòng)線程 -- 注意線程啟動(dòng)順序,啟動(dòng)順序很重要

????t2.start()

????t1.start()

?

????# 阻塞主線程,等待子線程結(jié)束

????t1.join()

????t2.join()

?

?

????print("程序結(jié)束!")

輸出結(jié)果:

1

2

3

4

5

6

7

8

A:成語接龍準(zhǔn)備好了嗎

B:準(zhǔn)備好了~開始吧!

A:一干二凈

B:凈你妹啊,沒法接...來個(gè)簡(jiǎn)單點(diǎn)的...

A:一天就知道看抖音美女,給你來個(gè)簡(jiǎn)單點(diǎn)的,來了:毛手毛腳

B:,這個(gè)我知道:腳踏實(shí)地

A:喲喲喲,不錯(cuò)不錯(cuò)!

程序結(jié)束!

python 線程條件變量Condition(31)

案例二:生產(chǎn)者與消費(fèi)者模式,以吃火鍋為例:一盤老肉片有10塊肉,吃完了又重新往鍋里加….

生產(chǎn)者:往鍋里加老肉片,每次加一盤(10塊);

消費(fèi)者:吃煮熟的肉片,沒吃一片,肉片數(shù)量減一,吃完為止;

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

# 導(dǎo)入線程模塊

import threading

import time

?

# 創(chuàng)建條件變量condition

con = threading.Condition()

meat_num = 0

?

def thread_consumers():

????# 條件變量condition 線程上鎖

????con.acquire()

????

????# 全局變量聲明關(guān)鍵字 global

????global meat_num

????meat_num = 0

?

????# 等待肉片下鍋煮熟

????con.wait()

????while True:

????????print("我來一塊肉片...")

????????meat_num -= 1

????????print("剩余肉片數(shù)量:%d"%meat_num)

????????time.sleep(0.5)

????????if meat_num == 0:

????????????# 肉片吃光了,通知老板添加肉片

????????????print("老板,再來一份老肉片...")

????????????con.notify()

????????????# 肉片吃光了,等待肉片

????????????con.wait()

?

????# 條件變量condition 線程釋放鎖

????con.release()

?

?

def thread_producer():

????# 條件變量condition 線程上鎖

????con.acquire()

????# 全局變量聲明關(guān)鍵字 global

????global meat_num

?

????# 肉片熟了,可以開始吃了

????meat_num = 10

????print("肉片熟了,可以開始吃了...")

????con.notify()

????while True:

????????# 阻塞函數(shù),等待肉片吃完的通知

????????con.wait()

????????meat_num = 10

????????# 添加肉片完成,可以繼續(xù)開吃

????????print("添加肉片成功!當(dāng)前肉片數(shù)量:%d"%meat_num)

????????time.sleep(1)

????????con.notify()

?

????con.release()

?

?

if __name__ == "__main__":

????# 創(chuàng)建并初始化線程

????t1 = threading.Thread(target=thread_producer)

????t2 = threading.Thread(target=thread_consumers)

?

????# 啟動(dòng)線程 -- 注意線程啟動(dòng)順序,啟動(dòng)順序很重要

????t2.start()

????t1.start()

?

????# 阻塞主線程,等待子線程結(jié)束

????t1.join()

????t2.join()

?

????print("程序結(jié)束!")

輸出結(jié)果:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

肉片熟了,可以開始吃了...

我來一塊肉片...

剩余肉片數(shù)量:9

我來一塊肉片...

剩余肉片數(shù)量:8

我來一塊肉片...

剩余肉片數(shù)量:7

我來一塊肉片...

剩余肉片數(shù)量:6

我來一塊肉片...

剩余肉片數(shù)量:5

我來一塊肉片...

剩余肉片數(shù)量:4

我來一塊肉片...

剩余肉片數(shù)量:3

我來一塊肉片...

剩余肉片數(shù)量:2

我來一塊肉片...

剩余肉片數(shù)量:1

我來一塊肉片...

剩余肉片數(shù)量:0

老板,再來一份老肉片...

添加肉片成功!當(dāng)前肉片數(shù)量:10

我來一塊肉片...

剩余肉片數(shù)量:9

我來一塊肉片...

剩余肉片數(shù)量:8

我來一塊肉片...

剩余肉片數(shù)量:7

.............

注意:

1.全局變量要聲明關(guān)鍵字?global;

2.注意線程的啟動(dòng)順序,這個(gè)很重要;

?

四.重點(diǎn)總結(jié)

注意線程互斥鎖Lock/線程事件Event/線程條件變量Condition三者的區(qū)別,場(chǎng)景不同,使用方式也不同,前兩者一般可以作為簡(jiǎn)單的線程交互,線程條件變量Condition可以用于比較復(fù)雜的線程交互!

猜你喜歡:

1.python線程創(chuàng)建和參數(shù)傳遞

2.python線程互斥鎖Lock

3.python線程事件Event

4.python return邏輯判斷表達(dá)式

?

轉(zhuǎn)載請(qǐng)注明:猿說Python???python條件變量Condition


向AI問一下細(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