我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Python中threading库实现线程锁与释放锁

短信预约 信息系统项目管理师 报名、考试、查分时间动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Python中threading库实现线程锁与释放锁

目录
  • 控制资源访问
  • 判断是否有另一个线程请求锁
  • with lock
  • 同步线程
    • Condition
    • 屏障(barrier)
  • 有限资源的并发访问
    • 隐藏资源

      控制资源访问

      前文提到threading库在多线程时,对同一资源的访问容易导致破坏与丢失数据。为了保证安全的访问一个资源对象,我们需要创建锁。

      示例如下:

      
      import threading
      import time
      
      class 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()

      运行之后,效果如下:

      解锁

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

      判断是否有另一个线程请求锁

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

      示例如下:

      
      import threading
      import time
      
      def 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()
      

      运行之后,效果如下:

      8次

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

      with lock

      前文,我们通过lock.acquire()与lock.release()实现了锁的获取与释放,但其实我们Python还给我们提供了一个更简单的语法,通过with lock来获取与释放锁。

      示例如下:

      
      import threading
      import time
      
      class 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 threading
      import time
      
      def 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 threading
      import time
      
      def 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 threading
      import time
      
      class 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 threading
      import random
      
      def 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 = 1000
      show_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对所有线程都不可见,除非在某个线程中设置了这个属性,这个线程才能看到它。

      到此这篇关于Python中threading库实现线程锁与释放锁的文章就介绍到这了,更多相关Python 线程锁与释放锁内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

      免责声明:

      ① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

      ② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

      Python中threading库实现线程锁与释放锁

      下载Word文档到电脑,方便收藏和打印~

      下载Word文档

      猜你喜欢

      Python中threading库实现线程锁与释放锁

      目录控制资源访问判断是否有另一个线程请求锁with lock同步线程Condition屏障(barrier)有限资源的并发访问隐藏资源控制资源访问 前文提到threading库在多线程时,对同一资源的访问容易导致破坏与丢失数据。为了保证安全
      2022-06-02

      Python中threading库如何实现线程锁与释放锁

      小编给大家分享一下Python中threading库如何实现线程锁与释放锁,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!控制资源访问前文提到threading库在
      2023-06-15

      python中怎么实现threading线程同步

      小编给大家分享一下python中怎么实现threading线程同步,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!说明1、threading模块具有实现锁定的内置功能,允许同步线程。为了防止数据损坏或丢失,需要锁定来控制共享
      2023-06-20

      python多线程中互斥锁与死锁的示例分析

      小编给大家分享一下python多线程中互斥锁与死锁的示例分析,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!一、多线程间的资源竞争以下列task1(),task2()两个函数为例,分别将对全局变量num加一重复一千万次循环(
      2023-06-29

      python中threading实现线程的示例分析

      小编给大家分享一下python中threading实现线程的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!过程说明1、从Thread类构成子类。覆盖方法根
      2023-06-20

      Python中GIL全局解释锁如何实现

      今天小编给大家分享一下Python中GIL全局解释锁如何实现的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。1.为什么有GIL
      2023-07-05

      Python实现的多线程同步与互斥锁功能示例

      本文实例讲述了Python实现的多线程同步与互斥锁功能。分享给大家供大家参考,具体如下:#! /usr/bin/env python #coding=utf-8 import threading import time ''' #1、不加锁
      2022-06-04

      Java编程之多线程死锁与线程间通信简单实现代码

      死锁定义 死锁是指两个或者多个线程被永久阻塞的一种局面,产生的前提是要有两个或两个以上的线程,并且来操作两个或者多个以上的共同资源;我的理解是用两个线程来举例,现有线程A和B同时操作两个共同资源a和b,A操作a的时候上锁LockA,继续执行
      2023-05-30

      Android中怎么利用同步锁实现多线程

      Android中怎么利用同步锁实现多线程,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。一、同步机制关键字synchronized 对于Java来说,最常用的同步机制就是sync
      2023-05-30

      简要讲解Python编程中线程的创建与锁的使用

      创建线程 创建线程的两种方法: 1,直接调用threading.Thread来构造thread对象,Thread的参数如下: class threading.Thread(group=None, target=None, name=None
      2022-06-04

      python实现多进程并发控制Semaphore与互斥锁LOCK

      一、了解锁应用场景举例描述: Lock 互斥锁:举例说明?有三个同事同时需要上厕所,但是只有一间厕所,将同事比作进程,多个进程并打抢占一个厕所,我们要保证顺序优先, 一个个来,那么就必须串行,先到先得,有人使用了,就锁住,结束后剩余的人继续
      2022-06-02

      Go语言sync包与锁实现限制线程对变量的访问

      本文主要介绍了Go语言sync包与锁实现限制线程对变量的访问,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
      2023-05-15

      Java中怎么利用多线程锁实现数据同步共享

      Java中怎么利用多线程锁实现数据同步共享,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。大多数应用程序要求线程互相通信来同步它们的动作。在Java程序中最简单实现同步的方法就是
      2023-06-17

      怎么使用Go语言sync包与锁实现限制线程对变量的访问

      本篇内容主要讲解“怎么使用Go语言sync包与锁实现限制线程对变量的访问”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“怎么使用Go语言sync包与锁实现限制线程对变量的访问”吧!Go语言中 sy
      2023-07-06

      python中websockets与主线程传递参数的实现

      Python中WebSockets与主线程传递参数可通过线程安全变量实现,如队列、事件信号和可重入锁。队列可用于传递数据,事件信号可用于发送信号触发操作,可重入锁可保护对共享数据的访问。使用这些方法可以安全且有效地实现WebSockets与主线程之间的参数传递。
      python中websockets与主线程传递参数的实现
      2024-04-02

      编程热搜

      • Python 学习之路 - Python
        一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
        Python 学习之路 - Python
      • chatgpt的中文全称是什么
        chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
        chatgpt的中文全称是什么
      • C/C++中extern函数使用详解
      • C/C++可变参数的使用
        可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
        C/C++可变参数的使用
      • css样式文件该放在哪里
      • php中数组下标必须是连续的吗
      • Python 3 教程
        Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
        Python 3 教程
      • Python pip包管理
        一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
        Python pip包管理
      • ubuntu如何重新编译内核
      • 改善Java代码之慎用java动态编译

      目录