首页 » Web前端 » phppthread释放技巧_threading库Python线程锁与释放锁

phppthread释放技巧_threading库Python线程锁与释放锁

访客 2024-11-28 0

扫一扫用手机浏览

文章目录 [+]

示例如下:

import threadingimport timeclass AddThread(): def __init__(self, start=0): self.lock = threading.Lock() self.value = start def increment(self): print("Wait Lock") self.lock.acquire() try: print("Acquire Lock") self.value += 1 print(self.value) finally: self.lock.release()def worker(a): time.sleep(1) a.increment()addThread = AddThread()for i in range(3): t = threading.Thread(target=worker, args=(addThread,)) t.start()

运行之后,效果如下:

phppthread释放技巧_threading库Python线程锁与释放锁

acquire()会通过锁进行壅塞其他线程实行中间段,release()开释锁,可以看到,基本都是得到锁之后才实行。
避免了多个线程同时改变其资源工具,不会造成混乱。

phppthread释放技巧_threading库Python线程锁与释放锁
(图片来自网络侵删)
判断是否有另一个线程要求锁

要确定是否有另一个线程要求锁而不影响当前的线程,可以设置acquire()的参数blocking=False。

示例如下:

import threadingimport timedef worker2(lock): print("worker2 Wait Lock") while True: lock.acquire() try: print("Holding") time.sleep(0.5) finally: print("not Holding") lock.release() time.sleep(0.5)def worker1(lock): print("worker1 Wait Lock") num_acquire = 0 value = 0 while num_acquire < 3: time.sleep(0.5) have_it = lock.acquire(blocking=False) try: value += 1 print(value) print("Acquire Lock") if have_it: num_acquire += 1 finally: print("release Lock") if have_it: lock.release()lock = threading.Lock()word2Thread = threading.Thread( target=worker2, name='work2', args=(lock,))word2Thread.start()word1Thread = threading.Thread( target=worker1, name='work1', args=(lock,))word1Thread.start()

运行之后,效果如下:

这里,我们须要迭代很多次,work1才能获取3次锁。
但是考试测验了8次。

with lock

前文,我们通过lock.acquire()与lock.release()实现了锁的获取与开释,但实在我们Python还给我们供应了一个更大略的语法,通过with lock来获取与开释锁。

示例如下:

import threadingimport timeclass AddThread(): def __init__(self, start=0): self.lock = threading.Lock() self.value = start def increment(self): print("Wait Lock") with self.lock: print("lock acquire") self.value += 1 print(self.value) print("lock release")def worker(a): time.sleep(1) a.increment()addThread = AddThread()for i in range(3): t = threading.Thread(target=worker, args=(addThread,)) t.start()

这里,我们只是将最上面的例子改变了一下。
效果如下:

须要把稳的是,正常的Lock工具不能要求多次,纵然是由同一个线程要求也不例外。
如果同一个调用链中的多个函数访问一个锁,则会发生意外。
如果期望在同一个线程的不同代码须要重新得到锁,那么这种情形下利用RLock。

同步线程Condition

在实际的操作中,我们还可以利用Condition工具来同步线程。
由于Condition利用了一个Lock,以是它可以绑定到一个共享资源,许可多个线程等待资源的更新。

示例如下:

import threadingimport timedef consumer(cond): print("waitCon") with cond: cond.wait() print('获取更新的资源')def producer(cond): print("worker") with cond: print('更新资源') cond.notifyAll()cond = threading.Condition()t1 = threading.Thread(name='t1', target=consumer, args=(cond,))t2 = threading.Thread(name='t2', target=consumer, args=(cond,))t3 = threading.Thread(name='t3', target=producer, args=(cond,))t1.start()time.sleep(0.2)t2.start()time.sleep(0.2)t3.start()

运行之后,效果如下:

这里,我们通过producer线程处理完成之后调用notifyAll(),consumer等线程等到了它的更新,可以类比为不雅观察者模式。
这里是,当一个线程用完资源之后时,则会自动关照依赖它的所有线程。

樊篱(barrier)

樊篱是另一种线程的同步机制。
barrier会建立一个掌握点,所有参与的线程会在这里壅塞,直到所有这些参与方都到达这一点。
采取这种方法,线程可以单独启动然后停息,直到所有线程都准备好了才可以连续。

示例如下:

import threadingimport timedef worker(barrier): print(threading.current_thread().getName(), "worker") worker_id = barrier.wait() print(threading.current_thread().getName(), worker_id)threads = []barrier = threading.Barrier(3)for i in range(3): threads.append( threading.Thread( name="t" + str(i), target=worker, args=(barrier,) ) )for t in threads: print(t.name, 'starting') t.start() time.sleep(0.1)for t in threads: t.join()

运行之后,效果如下:

从掌握台的输出会发创造,barrier.wait()会壅塞线程,直到所有线程被创建后,才同时开释超越这个掌握点连续实行。
wait()的返回值指示了开释的参与线程数,可以用来限定一些线程做清理资源等动作。

当然樊篱Barrier还有一个abort()方法,该方法可以使所有等待线程吸收一个BroKenBarrierError。
如果线程在wait()上被壅塞而停滞处理,会产生这个非常,通过except可以完成清理事情。

有限资源的并发访问

除了多线程可能访问同一个资源之外,有时候为了性能,我们也会限定多线程访问同一个资源的数量。
例如,线程池支持同时连接,但数据可能是固定的,或者一个网络APP供应的并发下载数支持固天命目。
这些连接就可以利用Semaphore来管理。

示例如下:

import threadingimport timeclass WorkerThread(threading.Thread): def __init__(self): super(WorkerThread, self).__init__() self.lock = threading.Lock() self.value = 0 def increment(self): with self.lock: self.value += 1 print(self.value)def worker(s, pool): with s: print(threading.current_thread().getName()) pool.increment() time.sleep(1) pool.increment()pool = WorkerThread()s = threading.Semaphore(2)for i in range(5): t = threading.Thread( name="t" + str(i), target=worker, args=(s, pool,) ) t.start()

运行之后,效果如下:

从图片虽然能看所有输出,但无法看到其停顿的韶光。
读者自己运行会创造,每次顶多只有两个线程在事情,是由于我们设置了threading.Semaphore(2)。

隐蔽资源

在实际的项目中,有些资源须要锁定以便于多个线程利用,而其余一些资源则须要保护,以使它们对并非使这些资源的所有者的线程隐蔽。

local()函数会创建一个工具,它能够隐蔽值,使其在不同的线程中无法被看到。
示例如下:

import threadingimport randomdef show_data(data): try: result = data.value except AttributeError: print(threading.current_thread().getName(), "No value") else: print(threading.current_thread().getName(), "value=", result)def worker(data): show_data(data) data.value = random.randint(1, 100) show_data(data)local_data = threading.local()show_data(local_data)local_data.value = 1000show_data(local_data)for i in range(2): t = threading.Thread( name="t" + str(i), target=worker, args=(local_data,) ) t.start()

运行之后,效果如下:

这里local_data.value对所有线程都不可见,除非在某个线程中设置了这个属性,这个线程才能看到它。

相关文章

介绍百度码,技术革新背后的智慧之光

随着科技的飞速发展,互联网技术已经成为我们生活中不可或缺的一部分。而在这个信息爆炸的时代,如何快速、准确地获取信息,成为了人们关注...

Web前端 2025-01-03 阅读1 评论0

介绍皮箱密码,开启神秘之门的钥匙

皮箱,作为日常生活中常见的收纳工具,承载着我们的珍贵物品。面对紧闭的皮箱,许多人却束手无策。如何才能轻松打开皮箱呢?本文将为您揭秘...

Web前端 2025-01-03 阅读1 评论0

介绍盗号器,网络安全的隐忧与应对步骤

随着互联网的快速发展,网络安全问题日益突出。盗号器作为一种非法工具,对网民的个人信息安全构成了严重威胁。本文将深入剖析盗号器的原理...

Web前端 2025-01-03 阅读1 评论0