python类库32[多线程同步Lock+RLock+Semaphore+Event]

 

多线程基础:python类库32[多线程同步]

 

一 多线程同步

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

 

完! 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Python线程编程中,锁(Lock)和可重入锁(RLock)都是常用的同步机制,用于保护共享资源,防止多个线程同时访问导致数据错误。 Lock是一种最基本的锁,它将资源锁住,直到锁被释放。当一个线程获得锁时,其他线程必须等待该线程释放锁后才能获得锁。这种锁是不可重入的,即同一个线程不能重复获得同一把锁。 RLock是可重入锁,它允许一个线程多次获得同一把锁。当一个线程获得锁时,它可以再次获得这个锁而不会被阻塞。只有该线程释放锁的次数与获得锁的次数相等时,其他线程才能获得该锁。可重入锁在需要多次获得同一把锁的场景中很有用。 下面是使用LockRLock的示例代码: ```python import threading # 创建一个Lock对象 lock = threading.Lock() # 创建一个RLock对象 rlock = threading.RLock() # 使用Lock保护共享资源 class Counter(object): def __init__(self): self.value = 0 def increment(self): lock.acquire() try: self.value += 1 finally: lock.release() # 使用RLock保护共享资源 class ReentrantCounter(object): def __init__(self): self.value = 0 def increment(self): rlock.acquire() try: self.value += 1 # 再次获得锁 rlock.acquire() try: self.value += 1 finally: rlock.release() finally: rlock.release() ``` 在上面的代码中,Counter类使用Lock保护value属性,而ReentrantCounter类使用RLock保护value属性。在increment方法中,Counter使用lock.acquire()和lock.release()获取和释放锁,在同一时间只允许一个线程访问value属性。而ReentrantCounter使用rlock.acquire()和rlock.release()获取和释放锁,并且在方法内部重复获得锁,这是RLock的特性。 需要注意的是,使用锁时要避免死锁的情况发生,即多个线程相互等待对方释放锁的情况。因此,在编写代码时要考虑好锁的获取和释放顺序,以避免死锁的发生。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值