当前位置: 移动技术网 > IT编程>脚本编程>Python > Python多线程同步Lock、RLock、Semaphore、Event实例

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

2019年06月14日  | 移动技术网IT编程  | 我要评论

王瑞儿轮奸门,金日成简介,和狗狗做的感觉

一、多线程同步

由于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()

结果:

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()

结果:

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")

运行结果:

三、其他

1) 线程局部变量

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

复制代码 代码如下:

mydata = threading.local()
mydata.x = 1

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

如对本文有疑问,请在下面进行留言讨论,广大热心网友会与你互动!! 点击进行留言回复

相关文章:

验证码:
移动技术网