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绘图方法实例入门
May 19 Python
python实现提取百度搜索结果的方法
May 19 Python
python中的break、continue、exit()、pass全面解析
Aug 05 Python
浅谈python函数之作用域(python3.5)
Oct 27 Python
TensorFlow实现iris数据集线性回归
Sep 07 Python
python实现五子棋小游戏
Mar 25 Python
Python datetime和unix时间戳之间相互转换的讲解
Apr 01 Python
python使用Plotly绘图工具绘制水平条形图
Mar 25 Python
python Django编写接口并用Jmeter测试的方法
Jul 31 Python
Matplotlib 绘制饼图解决文字重叠的方法
Jul 24 Python
python热力图实现简单方法
Jan 29 Python
PyTorch 实现L2正则化以及Dropout的操作
May 27 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
DC《神奇女侠2》因疫情推迟上映 温子仁新恐怖片《恶性》撤档
2020/04/09 欧美动漫
WHOIS类的修改版
2006/10/09 PHP
php strtotime 函数UNIX时间戳
2009/01/14 PHP
PHP 可阅读随机字符串代码
2010/05/26 PHP
php模拟socket一次连接,多次发送数据的实现代码
2011/07/26 PHP
php中XMLHttpRequest(Ajax)不能设置自定义的Referer的解决方法
2011/11/26 PHP
php版微信数据统计接口用法示例
2016/10/12 PHP
zTree插件之多选下拉菜单实例代码
2013/11/06 Javascript
禁用JavaScript控制台调试的方法
2014/03/07 Javascript
JavaScript获取function所有参数名的方法
2015/10/30 Javascript
实例解析jQuery插件EasyUI最常用的表单验证规则
2015/11/29 Javascript
jQuery处理XML文件的几种方法
2016/06/14 Javascript
js实现抽奖效果
2017/03/27 Javascript
ES6新数据结构Set与WeakSet用法分析
2017/03/31 Javascript
js下载文件并修改文件名
2017/05/08 Javascript
微信小程序实现顶部选项卡(swiper)
2020/06/19 Javascript
jQuery EasyUI结合zTree树形结构制作web页面
2017/09/01 jQuery
移动端网页开发调试神器Eruda的介绍与使用技巧
2017/10/30 Javascript
vue 做移动端微信公众号采坑经验记录
2018/04/26 Javascript
详解Node.js 中使用 ECDSA 签名遇到的坑
2018/11/26 Javascript
nodejs分离html文件里面的js和css的方法
2019/04/09 NodeJs
JS操作json对象key、value的常用方法分析
2019/10/29 Javascript
vue实现全屏滚动效果(非fullpage.js)
2020/03/07 Javascript
python获取mp3文件信息的方法
2015/06/15 Python
使用Python脚本将文字转换为图片的实例分享
2015/08/29 Python
Django应用程序中如何发送电子邮件详解
2017/02/04 Python
Python3 修改默认环境的方法
2019/02/16 Python
python-sys.stdout作为默认函数参数的实现
2020/02/21 Python
基于SpringBoot构造器注入循环依赖及解决方式
2020/04/26 Python
学习党课思想汇报
2013/12/29 职场文书
中学生期末评语
2014/02/03 职场文书
消防战士优秀事迹材料
2014/02/13 职场文书
电气自动化专业职业规划范文
2014/02/16 职场文书
大学运动会入场词
2014/02/22 职场文书
初中生散播谣言检讨书
2014/11/17 职场文书
2015年感恩节活动总结
2015/03/24 职场文书