上一篇随笔我们学了全局解释器锁,前面也学了互斥锁,今天学习一些与锁相关的点,例如递归锁,信号量,Event,还会学习我们已经很熟悉的队列,不过这次的队列是作为一个模块出现的。

本篇导航:

一、同步锁

1、join与互斥锁

线程抢的是GIL锁,GIL锁相当于执行权限,拿到执行权限后才能拿到互斥锁Lock,其他线程也可以抢到GIL,但如果发现Lock仍然没有被释放则阻塞,即便是拿到执行权限GIL也要立刻交出来

join是等待所有,即整体串行,而锁只是锁住修改共享数据的部分,即部分串行,要想保证数据安全的根本原理在于让并发变成串行,join与互斥锁都可以实现,毫无疑问,互斥锁的部分串行效率要更高

2、GIL VS Lock

锁的目的是为了保护共享的数据,同一时间只能有一个线程来修改共享的数据。结论:保护不同的数据就应该加不同的锁。

GIL 与Lock是两把锁,保护的数据不一样,前者是解释器级别的(当然保护的就是解释器级别的数据,比如垃圾回收的数据),后者是保护用户自己开发的应用程序的数据,很明显GIL不负责这件事,只能用户自定义加锁处理,即Lock

分析:

1)100个线程去抢GIL锁,即抢执行权限
2) 肯定有一个线程先抢到GIL(暂且称为线程1),然后开始执行,一旦执行就会拿到lock.acquire()
3)极有可能线程1还未运行完毕,就有另外一个线程2抢到GIL,然后开始运行,但线程2发现互斥锁lock还未被线程1释放,于是阻塞,被迫交出执行权限,即释放GIL
4)直到线程1重新抢到GIL,开始从上次暂停的位置继续执行,直到正常释放互斥锁lock,然后其他的线程再重复2 3 4的过程

3、join与互斥锁对比实例

1)未处理代码:

  1. #不加锁:并发执行,速度快,数据不安全
  2. from threading import currentThread,Thread
  3. import time
  4. def task():
  5. time.sleep(1)
  6. global n
  7. print('%s is running' %currentThread().getName())
  8. temp = n
  9. time.sleep(0.1)
  10. n = temp - 1
  11.  
  12. if __name__ == '__main__':
  13. n = 100
  14. t_l = []
  15. s1 = time.time()
  16. for i in range(100):
  17. t=Thread(target=task)
  18. t_l.append(t)
  19. t.start()
  20. for t in t_l:
  21. t.join()
  22.  
  23. s2 = time.time()
  24. print('主:%s n:%s' %(s2-s1,n))
  25.  
  26. '''
  27. Thread-1 is running
  28. Thread-2 is running
  29. ......
  30. Thread-100 is running
  31. 主:1.1128411293029785 n:99
  32. '''

初始

2)加互斥锁:

  1. #不加锁:未加锁部分并发执行,加锁部分串行执行,速度慢,数据安全
  2. from threading import currentThread,Thread,Lock
  3. import time
  4. def task():
  5. #未加锁的代码并发运行
  6. time.sleep(1)
  7. print('%s start to run' %currentThread().getName())
  8. global n
  9. #加锁的代码串行运行
  10. mutex.acquire()
  11. temp = n
  12. time.sleep(0.1)
  13. n = temp - 1
  14. mutex.release()
  15.  
  16. if __name__ == '__main__':
  17. n = 100
  18. mutex = Lock()
  19. t_l = []
  20. s1 = time.time()
  21. for i in range(100) :
  22. t = Thread(target=task)
  23. t_l.append(t)
  24. t.start()
  25. for t in t_l :
  26. t.join()
  27. s2 = time.time()
  28. print('主:%s n:%s' %(s2-s1,n))
  29.  
  30. '''
  31. Thread-1 is running
  32. Thread-2 is running
  33. ......
  34. Thread-100 is running
  35. 主:11.091605186462402 n:0
  36. '''

Lock

3)join效果

  1. from threading import currentThread,Thread
  2. import time
  3. def task():
  4. time.sleep(1)
  5. print('%s start to run' %currentThread().getName())
  6. global n
  7. temp = n
  8. time.sleep(0.1)
  9. n = temp - 1
  10.  
  11. if __name__ == '__main__':
  12. n = 100
  13. s1 = time.time()
  14. for i in range(100):
  15. t = Thread(target=task)
  16. t.start()
  17. t.join()
  18. s2 = time.time()
  19. print('主:%s n:%s' %(s2-s1,n))
  20.  
  21. '''
  22. Thread-1 start to run
  23. Thread-2 start to run
  24. ......
  25. Thread-100 start to run
  26. 主:110.16416668891907 n:0
  27. '''

join

即在start之后立刻使用jion,肯定会将100个任务的执行变成串行,毫无疑问,最终n的结果也肯定是0,是安全的,但问题是start后立即join:任务内的所有代码都是串行执行的,而加锁,只是加锁的部分即修改共享数据的部分是串行的单从保证数据安全方面,二者都可以实现,但很明显是加锁的效率更高.


二、死锁现象与递归锁

1、死锁现象

进程也有死锁与递归锁与线程中相同

死锁: 是指两个或两个以上的进程或线程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用,它们都将无法推进下去。此时称系统处于死锁状态或系统产生了死锁,这些永远在互相等待的进程称为死锁进程

  1. from threading import Lock,Thread
  2. import time
  3. mutexA=Lock()
  4. mutexB=Lock()
  5. class MyThread(Thread):
  6. def run(self):
  7. self.f1()
  8. self.f2()
  9.  
  10. def f1(self):
  11. mutexA.acquire()
  12. print('\033[31m%s 拿到A锁' %self.name)
  13. mutexB.acquire()
  14. print('\033[32m%s 拿到B锁' %self.name)
  15. mutexB.release()
  16. mutexA.release()
  17.  
  18. def f2(self):
  19. mutexB.acquire()
  20. print('\033[33m%s 拿到B锁' %self.name)
  21. time.sleep(1)
  22. mutexA.acquire()
  23. print('\033[34m%s 拿到A锁' %self.name)
  24. mutexA.release()
  25. mutexB.release()
  26.  
  27. if __name__ == '__main__':
  28. for i in range(10):
  29. t=MyThread()
  30. t.start()

死锁状态,程序永远无法结束:

2、递归锁

上述情况可以用递归锁解决

递归锁,在Python中为了支持在同一线程中多次请求同一资源,python提供了可重入锁RLock。

这个RLock内部维护着一个Lock和一个counter变量,counter记录了acquire的次数,从而使得资源可以被多次require。直到一个线程所有的acquire都被release,其他的线程才能获得资源。上面的例子如果使用RLock代替Lock,则不会发生死锁:

  1. from threading import ,Thread,RLock
  2. import time
  3. mutexB=mutexA=RLock()
  4. #一个线程拿到锁,counter加1,该线程内又碰到加锁的情况,则counter继续加1,这期间所有其他线程都只能等待,等待该线程释放所有锁,即counter递减到0为止
  5. class MyThread(Thread):
  6. def run(self):
  7. self.f1()
  8. self.f2()
  9.  
  10. def f1(self):
  11. mutexA.acquire()
  12. print('\033[31m%s 拿到A锁' %self.name)
  13. mutexB.acquire()
  14. print('\033[32m%s 拿到B锁' %self.name)
  15. mutexB.release()
  16. mutexA.release()
  17.  
  18. def f2(self):
  19. mutexB.acquire()
  20. print('\033[33m%s 拿到B锁' %self.name)
  21. time.sleep(1)
  22. mutexA.acquire()
  23. print('\033[34m%s 拿到A锁' %self.name)
  24. mutexA.release()
  25. mutexB.release()
  26.  
  27. if __name__ == '__main__':
  28. for i in range(10):
  29. t=MyThread()
  30. t.start()

三、信号量Semaphore

Semaphore也是一种锁不过这个锁可以自己定义同时可以进入锁的线程数

Semaphore管理一个内置的计数器,每当调用acquire()时内置计数器-1;调用release() 时内置计数器+1;

计数器不能小于0;当计数器为0时,acquire()将阻塞线程直到其他线程调用release()。

实例:

1、互斥锁Lock就像家里的厕所每次只能进一人,进去后锁门其他人在外面等着(这是学进程互斥锁时的例子)

  1. from multiprocessing import Process,Lock,current_process
  2. import time,random
  3. def work(mutex):
  4. mutex.acquire() #上锁
  5. print('%s 上厕所' %current_process().name)
  6. time.sleep(random.randint(1,3))
  7. print('%s 走了' %current_process().name)
  8. mutex.release() #开锁
  9.  
  10. if __name__ == '__main__':
  11. mutex=Lock() #实例化(互斥锁)
  12. print('start...')
  13. for i in range(20):
  14. t=Process(target=work,args=(mutex,))
  15. t.start()

2、信号量Semaphore就像是街道的公共厕所有固定个数的隔间(例如5个),刚开始可以进去5个,然后出来几个便可以再进去几个

  1. from threading import Thread,Semaphore,currentThread
  2. import time,random
  3. sm=Semaphore(5)
  4. def task():
  5. sm.acquire()
  6. print('%s 上厕所' %currentThread().getName())
  7. time.sleep(random.randint(1,3))
  8. print('%s 走了' %currentThread().getName())
  9. sm.release()
  10. if __name__ == '__main__':
  11. for i in range(20):
  12. t=Thread(target=task)
  13. t.start()

与进程池相似但是完全不同的概念,进程池Pool(4),最大只能产生4个进程,而且从头到尾都只是这四个进程,不会产生新的,而信号量是产生一堆线程/进程


四、Event

线程的一个关键特性是每个线程都是独立运行且状态不可预测。如果程序中的其 他线程需要通过判断某个线程的状态来确定自己下一步的操作,这时线程同步问题就会变得非常棘手。为了解决这些问题,我们需要使用threading库中的Event对象。 对象包含一个可由线程设置的信号标志,它允许线程等待某些事件的发生。在 初始情况下,Event对象中的信号标志被设置为假。如果有线程等待一个Event对象, 而这个Event对象的标志为假,那么这个线程将会被一直阻塞直至该标志为真。一个线程如果将一个Event对象的信号标志设置为真,它将唤醒所有等待这个Event对象的线程。如果一个线程等待一个已经被设置为真的Event对象,那么它将忽略这个事件, 继续执行

  1. event.isSet():返回event的状态值;
  2. event.wait():如果 event.isSet()==False将阻塞线程;
  3. event.set(): 设置event的状态值为True,所有阻塞池的线程激活进入就绪状态, 等待操作系统调度;
  4. event.clear():恢复event的状态值为False

1、模拟红绿灯

  1. from threading import Thread,Event,currentThread
  2. import time
  3. e=Event()
  4.  
  5. def traffic_lights():
  6. time.sleep(5)
  7. e.set()
  8.  
  9. def car():
  10. print('\033[41m%s 等' %currentThread().getName())
  11. e.wait()
  12. print('\033[42m%s 跑' %currentThread().getName())
  13.  
  14. if __name__ == '__main__':
  15. for i in range(10):
  16. t=Thread(target=car)
  17. t.start()
  18. traffic_thread=Thread(target=traffic_lights)
  19. traffic_thread.start()

2、有多个工作线程尝试链接MySQL,我们想要在链接前确保MySQL服务正常才让那些工作线程去连接MySQL服务器,如果连接不成功,都会去尝试重新连接。那么我们就可以采用threading.Event机制来协调各个工作线程的连接操作

  1. from threading import Thread,Event,currentThread
  2. import time
  3. e=Event()
  4. def conn_mysql():
  5. count=1
  6. while not e.is_set():
  7. if count > 3:
  8. raise ConnectionError('尝试链接的次数过多')
  9. print('\033[35m%s 第%s次尝试' %(currentThread().getName(),count))
  10. e.wait(timeout=1)
  11. count+=1
  12. print('\033[32m%s 开始链接' %currentThread().getName())
  13.  
  14. def check_mysql():
  15. print('\033[34m%s 检测mysql...' %currentThread().getName())
  16. time.sleep(2)
  17. e.set()
  18. if __name__ == '__main__':
  19. for i in range(3):
  20. t=Thread(target=conn_mysql)
  21. t.start()
  22. t=Thread(target=check_mysql)
  23. t.start()

五、定时器

定时器,指定n秒后执行某操作

  1. from threading import Timer
  2.  
  3. def hello(n):
  4. print("hello, world",n)
  5.  
  6. #三秒后运行hello函数传入参数123
  7. t = Timer(3, hello, args=(123,))
  8. t.start()

六、线程queue

queue队列 :使用import queue,用法与进程Queue一样

queue is especially useful in threaded programming when information must be exchanged safely between multiple threads.

队列在线程编程中尤其有用,因为必须在多个线程之间安全地交换信息。

1、queue.Queue() 先进先出

  1. import queue
  2.  
  3. q=queue.Queue()
  4. q.put('first')
  5. q.put('second')
  6. q.put('third')
  7.  
  8. print(q.get())
  9. print(q.get())
  10. print(q.get())
  11. '''
  12. 结果(先进先出):
  13. first
  14. second
  15. third
  16. '''

2、queue.LifoQueue() 后进先出

  1. import queue
  2.  
  3. q=queue.LifoQueue()
  4. q.put('first')
  5. q.put('second')
  6. q.put('third')
  7.  
  8. print(q.get())
  9. print(q.get())
  10. print(q.get())
  11. '''
  12. 结果(后进先出):
  13. third
  14. second
  15. first
  16. '''

3、queue.PriorityQueue() 存储数据时可设置优先级的队列

  1. import queue
  2.  
  3. q=queue.PriorityQueue()
  4. #put进入一个元组,元组的第一个元素是优先级(通常是数字,也可以是非数字之间的比较),数字越小优先级越高
  5. q.put((20,'a'))
  6. q.put((10,'b'))
  7. q.put((30,'c'))
  8.  
  9. print(q.get())
  10. print(q.get())
  11. print(q.get())
  12. '''
  13. 结果(数字越小优先级越高,优先级高的优先出队):
  14. (10, 'b')
  15. (20, 'a')
  16. (30, 'c')
  17. '''

34、锁问题与线程queue的更多相关文章

  1. python开发线程:死锁和递归锁&信号量&定时器&线程queue&事件evevt

    一 死锁现象与递归锁 进程也有死锁与递归锁,在进程那里忘记说了,放到这里一切说了额 所谓死锁: 是指两个或两个以上的进程或线程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用,它们都将 ...

  2. 锁问题与线程queue

    一.同步锁 1.join与互斥锁 线程抢的是GIL锁,GIL锁相当于执行权限,拿到执行权限后才能拿到互斥锁Lock,其他线程也可以抢到GIL,但如果发现Lock仍然没有被释放则阻塞,即便是拿到执行权限 ...

  3. python并发编程之线程(二):死锁和递归锁&信号量&定时器&线程queue&事件evevt

    一 死锁现象与递归锁 进程也有死锁与递归锁,在进程那里忘记说了,放到这里一切说了额 所谓死锁: 是指两个或两个以上的进程或线程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用,它们都将 ...

  4. 死锁与递归锁 信号量 event 线程queue

    1.死锁现象与递归锁 死锁:是指两个或两个以上的进程或线程在执行过程中,因争抢资源而造成的一种互相等待的现象,若无外力作用,它们都将无法推进下去,此时称系统处于死锁状态或系统产生了死锁,这些永远在互相 ...

  5. Python之网路编程之死锁,递归锁,信号量,Event事件,线程Queue

    一.死锁现象与递归锁 进程也是有死锁的 所谓死锁: 是指两个或两个以上的进程或线程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用, 它们都将无法推进下去.此时称系统处于死锁状态或系统 ...

  6. 1.gil全局解释器锁, 2. 死锁与递归锁 3. 信号量 4. Event事件 5. 线程queue

    gil本质就是一把互斥锁,相当于执行权限,每个进程都会存在一把gil,同一进程内的多个线程必须抢到gil 之后才能使用cpython解释器来执行自己的代码,同一进程下的多线程不能并行,但可以实现并发 ...

  7. Python 36 死锁现象和递归锁、信号量、Event事件、线程queue

    一:死锁现象和递归锁 所谓死锁: 是指两个或两个以上的进程或线程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用,它们都将无法推进下去.此时称系统处于死锁状态或系统产生了死锁,这些永远 ...

  8. GIL全局解释锁,死锁,信号量,event事件,线程queue,TCP服务端实现并发

    一.GIL全局解释锁 在Cpython解释器才有GIL的概念,不是python的特点 在Cpython解释器中,同一个进程下开启的多线程,同一时刻只能有一个线程执行,无法利用多核优势. 1.GIL介绍 ...

  9. 同步锁 死锁与递归锁 信号量 线程queue event事件

    二个需要注意的点: 1 线程抢的是GIL锁,GIL锁相当于执行权限,拿到执行权限后才能拿到互斥锁Lock,其他线程也可以抢到GIL,但如果发现Lock任然没有被释放则阻塞,即便是拿到执行权限GIL也要 ...

随机推荐

  1. css3变换,过度,动画实现梦幻网页

    html和css3一出,整个互联网设计发生了颠覆性的改变,各大IT企业也推出了很多新颖的设计,比如百度浏览器的下载首页,fullpage设计风格加css动画让网页看起来很流畅舒服. css3的变换有3 ...

  2. 平方根的C语言实现(一)

    曾经做一个硬件成本极度控制的项目,因为硬件成本极低,并且还需要实现较高的精度测量,过程中也自己用C语言实现了正弦.余弦.反正切.平方根等函数. 以下,无论是在我的实际项目中还是本地的计算机系统,int ...

  3. 可存放任意类型变量的动态数组--C语言实现

    之前在训练营的时候被要求用C语言实现一个可以存放任意类型数据的栈.现在尝试实现一个数组版本. 首先用到的结构体如下(接触了Win32编程所以长得有点像里面的那些类型): typedef struct ...

  4. (转)FastJson---高性能JSON开发包

    场景:javaBean对象转化为json对象! 1 Fastjson介绍 Fastjson是一个Java语言编写的JSON处理器,由阿里巴巴公司开发.1.遵循http://json.org标准,为其官 ...

  5. 白话ASP.NET MVC之一:Url 路由

    好久没有写关于ASP.NET MVC的东西了,虽然<ASP.NET MVC4框架揭秘>已经完完整整的看完一遍,但是感觉和一锅粥差不多,没什么可写的,因为我自己不理解,也就写不出来.现在开始 ...

  6. 如何将R包安装到自定义路径

    参考  设置环境变量R_LIBS将R包安装到自定义路径   实际上是可以解决问题的, #环境变量完成以后,启动(重启)R,运行 .libPaths() 加载R包时,发现路径仍然未变成自定义的. 那么参 ...

  7. java网络编程实现两端聊天

    网络编程的三要素: ip地址:唯一标识网络上的每一台计算机 端口号:计算机中应用的标号(代表一个应用程序),0-1024系统使用或者保留端口,有效端口0-65535(short) 通信协议:通信的规则 ...

  8. UIButton 中高亮取消

    1.图片 取消高亮方法 a.第一种方法 [withdrawalBtn setAdjustsImageWhenHighlighted:NO]; b.第二种方法 [withdrawalBtn setIma ...

  9. Entity Framework Core 2.0 新特性

    本文翻译来自:https://docs.microsoft.com/en-us/ef/core/what-is-new/index 一.模型级查询过滤器(Model-level query filte ...

  10. 安装完iis后本机用IP可以访问,别的电脑不能访问

    局域网IIS无法访问的解决方法 无法访问局域网内其它电脑中的IIS,自己可以访问,但别的电脑不行一般发生这种情况时操作系统是XP第二版. 具体操作如下:1.打开控制面版2.打开Windows防火墙 5 ...