锁,Event,semaphore
GIL:全局解释锁:无论开启多少个线程,同一时刻只允许执行一个线程运行(解释器级别,保护数据)
锁:两种状态,锁定和未锁定,仅支持两个函数,获得锁和释放锁
多线程抢夺锁时,当某个线程获得锁了,其他的锁都会被阻塞,直到这个线程释放锁后,其他线程才可以再抢夺锁,注意线程抢夺锁是没有顺序的
同步锁:Lock,与GIL不同,它是用户层面上的,由用户自定义
import threading
import time def sub():
global num # 加上一个锁,使得本来并发的变成串行,用来保护自己的数据不被其他的线程拿到
lock.acquire() #获得锁
temp = num
time.sleep(0.01) #当阻碍时间较长时,其他的线程可能也会拿到这个数据
num = temp - 1
lock.release() #释放锁 num = 100 l = []
lock = threading.Lock() #声明一个锁 for i in range(100):
t = threading.Thread(target=sub)
t.start()
l.append(t)
for t in l:
t.join()
print(num)
死锁:当存在多线程时,双方都在等对方释放锁,产生死锁,当无外力作用时,程序将无法推进下去
递归锁:RLock内部维护着一个counter,每次acquire都会使得counter + 1 ,每release一次counter - 1,
当counter大于0时,只能有一个线程可以获得这个锁,只有等于0时,其他线程才能获得这个锁
import threading
import time class MyThread(threading.Thread):
def action1(self):
t1.acquire()
print(self.name,"got t1",time.ctime)
time.sleep(2) t2.acquire()
print(self.name,"got t2",time.ctime)
time.sleep(2) t2.release()
t1.release() def action2(self):
t2.acquire()
print(self.name, "got t2", time.ctime)
time.sleep(2) t1.acquire()
print(self.name, "got t1", time.ctime)
time.sleep(2) t1.release()
t2.release() def run(self):
self.action1()
self.action2() if __name__ == '__main__':
t1 = threading.Lock()
t2 = threading.Lock() L = [] for i in range(5):
t = MyThread()
t.start()
L.append(t) for t in L:
t.join() print("end...")
解决产生死锁:使用递归锁,t1,t2的锁声明改为 threading.RLock()
同步条件:Event
#创建Event对象
event = threading.Event 3个方法
#a client thread wait for the flag to be set
event.wait() #a server thread can set or reset it
event.set()
event.clear()
使用规则
if the flag is set , the wait method doesn't do anything
if the flag is cleared ,the wait block until it becomes set again
any number of threads may wait for the same event
import threading,time
class Boss(threading.Thread):
def run(self):
print("BOSS:everybody works overtime tonight。")
print(event.isSet())
event.set()
time.sleep(5)
print("BOSS:can leave work。")
print(event.isSet())
event.set()
class Worker(threading.Thread):
def run(self):
event.wait() #当Event没有set时,wait方法被调用,阻塞,当Event被set后,这个wait方法就相当于pass
print("Worker:ah……life is bitter!")
time.sleep(1)
event.clear()
event.wait()
print("Worker:OhYeah!")
if __name__=="__main__":
event=threading.Event()
threads=[]
for i in range(5):
threads.append(Worker())
threads.append(Boss())
for t in threads:
t.start()
for t in threads:
t.join()
信号量:Semaphore
用来控制线程并发数,当同时并发的线程达到指定的最大数时,其他线程将会被阻塞
BoundedSemaphore或Semaphore内置一个计数器,当acquire()时 -1,当release()时 +1,
计数器不能小于0,当计数器等于0时,acquire()将线程阻塞同步至锁定状态,直到有线程release()释放
BoundedSemaphore和Semaphore的区别:BoundedSemaphore在调用release()检测计数器是否超过设定值,达到抛出一个异常
import threading,time
class myThread(threading.Thread):
def run(self):
if semaphore.acquire():
print(self.name)
time.sleep(5)
semaphore.release()
if __name__=="__main__":
semaphore=threading.Semaphore(5) #设置可以同时有几个线程并发
thrs=[]
for i in range(100):
thrs.append(myThread())
for t in thrs:
t.start()
锁,Event,semaphore的更多相关文章
- 【分布式锁】07-Zookeeper实现分布式锁:Semaphore、读写锁实现原理
前言 前面已经讲解了Zookeeper可重入锁的实现原理,自己对分布式锁也有了更深的认知. 我在公众号中发了一个疑问,相比于Redis来说,Zookeeper的实现方式要更好一些,即便Redis作者实 ...
- Day12- Python基础12 线程、GIL、Lock锁、RLock锁、Semaphore锁、同步条件event
http://www.cnblogs.com/yuanchenqi/articles/6248025.html 博客地址 本节内容: 1:进程和线程的说明 2:线程的两种调用方式 3:threadi ...
- 并发编程~~~多线程~~~守护线程, 互斥锁, 死锁现象与递归锁, 信号量 (Semaphore), GIL全局解释器锁
一 守护线程 from threading import Thread import time def foo(): print(123) time.sleep(1) print('end123') ...
- 带你看看Java的锁(二)-Semaphore
前言 简介 Semaphore 中文称信号量,它和ReentrantLock 有所区别,ReentrantLock是排他的,也就是只能允许一个线程拥有资源,Semaphore是共享的,它允许多个线程同 ...
- 同步锁 死锁与递归锁 信号量 线程queue event事件
二个需要注意的点: 1 线程抢的是GIL锁,GIL锁相当于执行权限,拿到执行权限后才能拿到互斥锁Lock,其他线程也可以抢到GIL,但如果发现Lock任然没有被释放则阻塞,即便是拿到执行权限GIL也要 ...
- Python之网路编程之死锁,递归锁,信号量,Event事件,线程Queue
一.死锁现象与递归锁 进程也是有死锁的 所谓死锁: 是指两个或两个以上的进程或线程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用, 它们都将无法推进下去.此时称系统处于死锁状态或系统 ...
- Python 中Semaphore 信号量对象、Event事件、Condition
Semaphore 信号量对象 信号量是一个更高级的锁机制.信号量内部有一个计数器而不像锁对象内部有锁标识,而且只有当占用信号量的线程数超过信号量时线程才阻塞.这允许了多个线程可以同时访问相同的代码区 ...
- python基础-12 多线程queue 线程交互event 线程锁 自定义线程池 进程 进程锁 进程池 进程交互数据资源共享
Python中的进程与线程 学习知识,我们不但要知其然,还是知其所以然.你做到了你就比别人NB. 我们先了解一下什么是进程和线程. 进程与线程的历史 我们都知道计算机是由硬件和软件组成的.硬件中的CP ...
- python多线程锁lock/Rlock/BoundedSemaphore/Condition/Event
import time import threading lock = threading.RLock() n = 10 def task(arg): # 加锁,此区域的代码同一时刻只能有一个线程执行 ...
- 34、锁问题与线程queue
上一篇随笔我们学了全局解释器锁,前面也学了互斥锁,今天学习一些与锁相关的点,例如递归锁,信号量,Event,还会学习我们已经很熟悉的队列,不过这次的队列是作为一个模块出现的. 一.同步锁 1.join ...
随机推荐
- 【Django】搭建Django administration并登录
Python自带一个后台管理系统,这个后台管理系统搭建与登陆呢? 新建项目Django_Admin 文件结构目录如下: 创建APP 在pycharm下方的terminal终端中输入命令: python ...
- spring cloud:gateway-eureka-filter
Spring Cloud Gateway 的 Filter 的生命周期不像 Zuul 的那么丰富,它只有两个:“pre” 和 “post”. PRE: 这种过滤器在请求被路由之前调用.我们可利用这种过 ...
- eclipse连接Mysql和测试
一.前期准备: 1.eclipse 2.Mysql workbench 3.jdbc 下载地址:https://www.mysql.com/products/connector/ 点击JDBC Dri ...
- EventBus-实现java状态机
摘自:https://www.jianshu.com/p/8def04b34b3c 首先,了解状态机是什么,我们为什么需要状态机! 举个最简单例子,请假,作为一个最底层程序员,每次请假都要领导层层审批 ...
- php中处理汉字字符串长度:strlen和mb_strlen
PHP内置的字符串长度函数strlen()无法正确处理中文字符串,它得到的只是字符串所占的字节数.对于GB2312的中文编码,strlen得到的值是汉字个数的2倍,而对于UTF-8编码的中文,就是3倍 ...
- 数据结构和算法(Java版)快速学习(线性表)
线性表的基本特征: 第一个数据元素没有前驱元素: 最后一个数据元素没有后继元素: 其余每个数据元素只有一个前驱元素和一个后继元素. 线性表按物理存储结构的不同可分为顺序表(顺序存储)和链表(链式存储) ...
- 3D聚类
1 3D聚类和普通的二维聚类实质一样,只不过维数太高了,用三维图来表示了. 下面将官网的改成只生成一个图了 #!/usr/bin/python # -*- coding:utf-8 -*- print ...
- CSS3——边框 圆角 背景 渐变 文本效果
边框 圆角边框 盒阴影 边界图片 圆角 CSS3 圆角制作器 指定每个角 背景 多重背景图像 大小 图像的定位 背景剪裁 渐变 线性渐变(Linear Gradients)- 向下/向上/向左/向右/ ...
- 【HANA系列】SAP HANA SQL获取当前日期
公众号:SAP Technical 本文作者:matinal 原文出处:http://www.cnblogs.com/SAPmatinal/ 原文链接:[HANA系列]SAP HANA SQL获取当前 ...
- 【MM系列】SAP 物料进销存报表查看
公众号:SAP Technical 本文作者:matinal 原文出处:http://www.cnblogs.com/SAPmatinal/ 原文链接:[MM系列]在SAP里查看数据的方法 前言部 ...