Python多线程同步Lock、RLock、Semaphore、Event实例


Posted in Python onNovember 21, 2014

一、多线程同步

由于CPython的python解释器在单线程模式下执行,所以导致python的多线程在很多的时候并不能很好地发挥多核cpu的资源。大部分情况都推荐使用多进程。

python的多线程的同步与其他语言基本相同,主要包含:

Lock & RLock :用来确保多线程多共享资源的访问。
Semaphore : 用来确保一定资源多线程访问时的上限,例如资源池。 
Event : 是最简单的线程间通信的方式,一个线程可以发送信号,其他的线程接收到信号后执行操作。 

二、实例

1)Lock & RLock

Lock对象的状态可以为locked和unlocked

使用acquire()设置为locked状态;
使用release()设置为unlocked状态。

如果当前的状态为unlocked,则acquire()会将状态改为locked然后立即返回。当状态为locked的时候,acquire()将被阻塞直到另一个线程中调用release()来将状态改为unlocked,然后acquire()才可以再次将状态置为locked。

Lock.acquire(blocking=True, timeout=-1),blocking参数表示是否阻塞当前线程等待,timeout表示阻塞时的等待时间 。如果成功地获得lock,则acquire()函数返回True,否则返回False,timeout超时时如果还没有获得lock仍然返回False。

实例:(确保只有一个线程可以访问共享资源)

import threading

import time

 

num = 0

lock = threading.Lock()

 

def func(st):

    global num

    print (threading.currentThread().getName() + ' try to acquire the lock')

    if lock.acquire():

        print (threading.currentThread().getName() + ' acquire the lock.' )

        print (threading.currentThread().getName() +" :%s" % str(num) )

        num += 1

        time.sleep(st)

        print (threading.currentThread().getName() + ' release the lock.'  )        

        lock.release()

 

t1 = threading.Thread(target=func, args=(8,))

t2 = threading.Thread(target=func, args=(4,))

t3 = threading.Thread(target=func, args=(2,))

t1.start()

t2.start()

t3.start()

结果:

Python多线程同步Lock、RLock、Semaphore、Event实例

RLock与Lock的区别是:RLock中除了状态locked和unlocked外还记录了当前lock的owner和递归层数,使得RLock可以被同一个线程多次acquire()。

2)Semaphore

Semaphore管理一个内置的计数器,
每当调用acquire()时内置计数器-1;
调用release() 时内置计数器+1;
计数器不能小于0;当计数器为0时,acquire()将阻塞线程直到其他线程调用release()。

实例:(同时只有2个线程可以获得semaphore,即可以限制最大连接数为2):

import threading

import time
semaphore = threading.Semaphore(2)

 

def func():

    if semaphore.acquire():

        for i in range(5):

          print (threading.currentThread().getName() + ' get semaphore')

        semaphore.release()

        print (threading.currentThread().getName() + ' release semaphore')

        

        

for i in range(4):

  t1 = threading.Thread(target=func)

  t1.start()

结果:

Python多线程同步Lock、RLock、Semaphore、Event实例

3) Event

Event内部包含了一个标志位,初始的时候为false。
可以使用使用set()来将其设置为true;
或者使用clear()将其从新设置为false;
可以使用is_set()来检查标志位的状态;
另一个最重要的函数就是wait(timeout=None),用来阻塞当前线程,直到event的内部标志位被设置为true或者timeout超时。如果内部标志位为true则wait()函数理解返回。

实例: (线程间相互通信)

import logging

import threading

import time
logging.basicConfig(level=logging.DEBUG,

format="(%(threadName)-10s : %(message)s",

)
def wait_for_event_timeout(e, t):

    """Wait t seconds and then timeout"""

    while not e.isSet():

      logging.debug("wait_for_event_timeout starting")

      event_is_set = e.wait(t)

      logging.debug("event set: %s" % event_is_set)

    if event_is_set:

      logging.debug("processing event")

    else:

      logging.debug("doing other work")

      

e = threading.Event()

t2 = threading.Thread(name="nonblock",

target=wait_for_event_timeout,args=(e, 2))

t2.start()

logging.debug("Waiting before calling Event.set()")

time.sleep(7)

e.set()

logging.debug("Event is set")

运行结果:

Python多线程同步Lock、RLock、Semaphore、Event实例

三、其他

1) 线程局部变量

线程局部变量的值是跟线程相关的,区别与全局的变量。使用非常简单如下:

mydata = threading.local()

mydata.x = 1

2)对Lock,semaphore,condition等使用with关键字代替手动调用acquire()和release()。

Python 相关文章推荐
简单讲解Python中的数字类型及基本的数学计算
Mar 11 Python
Python 的内置字符串方法小结
Mar 15 Python
Python实现的简单计算器功能详解
Aug 25 Python
Python 窗体(tkinter)按钮 位置实例
Jun 13 Python
python turtle库画一个方格和圆实例
Jun 27 Python
Python实现FLV视频拼接功能
Jan 21 Python
Python3之外部文件调用Django程序操作model等文件实现方式
Apr 07 Python
django创建超级用户时指定添加其它字段方式
May 14 Python
Python命名空间及作用域原理实例解析
Aug 12 Python
python中如何使用虚拟环境
Oct 14 Python
在终端启动Python时报错的解决方案
Nov 20 Python
python将下载到本地m3u8视频合成MP4的代码详解
Nov 24 Python
python多进程操作实例
Nov 21 #Python
Python多进程通信Queue、Pipe、Value、Array实例
Nov 21 #Python
Python多进程同步Lock、Semaphore、Event实例
Nov 21 #Python
Python multiprocessing.Manager介绍和实例(进程间共享数据)
Nov 21 #Python
Python pickle类库介绍(对象序列化和反序列化)
Nov 21 #Python
Python和perl实现批量对目录下电子书文件重命名的代码分享
Nov 21 #Python
Python实现的下载8000首儿歌的代码分享
Nov 21 #Python
You might like
用PHP和ACCESS写聊天室(五)
2006/10/09 PHP
如何使用Gitblog和Markdown建自己的博客
2015/07/31 PHP
详解WordPress中分类函数wp_list_categories的使用
2016/01/04 PHP
Laravel 5.5 实现禁用用户注册示例
2019/10/24 PHP
javascript下IE与FF兼容函数收集
2008/09/17 Javascript
javascript onkeydown,onkeyup,onkeypress,onclick,ondblclick
2009/02/04 Javascript
JavaScript 加号(+)运算符号
2009/12/06 Javascript
javascript 模式设计之工厂模式学习心得
2010/04/27 Javascript
JS高级笔记
2011/07/13 Javascript
js函数返回多个返回值的示例代码
2013/11/05 Javascript
JS实现仿京东淘宝竖排二级导航
2014/12/08 Javascript
JavaScript作用域链实例详解
2019/01/21 Javascript
如何在Node和浏览器控制台中打印彩色文字
2020/01/09 Javascript
微信小程序实现打卡签到页面
2020/09/21 Javascript
横向对比分析Python解析XML的四种方式
2016/03/30 Python
你应该知道的python列表去重方法
2017/01/17 Python
浅谈对yield的初步理解
2017/05/29 Python
Sanic框架配置操作分析
2018/07/17 Python
Python实现数值积分方式
2019/11/20 Python
PyTorch笔记之scatter()函数的使用
2020/02/12 Python
python对execl 处理操作代码
2020/06/22 Python
推荐WEB开发者最佳HTML5和CSS3代码生成器
2015/11/24 HTML / CSS
给朋友的道歉信
2014/01/09 职场文书
办公室副主任职责范本
2014/03/08 职场文书
给小学生的新年寄语
2014/04/04 职场文书
分家协议书
2014/04/21 职场文书
新文化运动的口号
2014/06/21 职场文书
美术专业自荐信
2014/07/07 职场文书
运动会广播稿150字(9篇)
2014/09/20 职场文书
班子查摆四风个人对照检查材料思想汇报
2014/10/04 职场文书
道德与公民自我评价
2015/03/09 职场文书
2015年后勤工作总结范文
2015/04/08 职场文书
南京大屠杀观后感
2015/06/02 职场文书
2015年计算机教师工作总结
2015/07/22 职场文书
Python中可变和不可变对象的深入讲解
2021/08/02 Python
利用Python实时获取steam特惠游戏数据
2022/06/25 Python