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的更多相关文章

  1. 【分布式锁】07-Zookeeper实现分布式锁:Semaphore、读写锁实现原理

    前言 前面已经讲解了Zookeeper可重入锁的实现原理,自己对分布式锁也有了更深的认知. 我在公众号中发了一个疑问,相比于Redis来说,Zookeeper的实现方式要更好一些,即便Redis作者实 ...

  2. Day12- Python基础12 线程、GIL、Lock锁、RLock锁、Semaphore锁、同步条件event

    http://www.cnblogs.com/yuanchenqi/articles/6248025.html  博客地址 本节内容: 1:进程和线程的说明 2:线程的两种调用方式 3:threadi ...

  3. 并发编程~~~多线程~~~守护线程, 互斥锁, 死锁现象与递归锁, 信号量 (Semaphore), GIL全局解释器锁

    一 守护线程 from threading import Thread import time def foo(): print(123) time.sleep(1) print('end123') ...

  4. 带你看看Java的锁(二)-Semaphore

    前言 简介 Semaphore 中文称信号量,它和ReentrantLock 有所区别,ReentrantLock是排他的,也就是只能允许一个线程拥有资源,Semaphore是共享的,它允许多个线程同 ...

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

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

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

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

  7. Python 中Semaphore 信号量对象、Event事件、Condition

    Semaphore 信号量对象 信号量是一个更高级的锁机制.信号量内部有一个计数器而不像锁对象内部有锁标识,而且只有当占用信号量的线程数超过信号量时线程才阻塞.这允许了多个线程可以同时访问相同的代码区 ...

  8. python基础-12 多线程queue 线程交互event 线程锁 自定义线程池 进程 进程锁 进程池 进程交互数据资源共享

    Python中的进程与线程 学习知识,我们不但要知其然,还是知其所以然.你做到了你就比别人NB. 我们先了解一下什么是进程和线程. 进程与线程的历史 我们都知道计算机是由硬件和软件组成的.硬件中的CP ...

  9. python多线程锁lock/Rlock/BoundedSemaphore/Condition/Event

    import time import threading lock = threading.RLock() n = 10 def task(arg): # 加锁,此区域的代码同一时刻只能有一个线程执行 ...

  10. 34、锁问题与线程queue

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

随机推荐

  1. Java缓存机制

    1 Java缓存 1.1 jvm内置缓存 Java中实现缓存的方式有很多,比如用static hashMap基于内存缓存的jvm内置缓存,简单不实用,保对象的有效性和周期无法控制,容易造成内存急剧上升 ...

  2. 【Python】学习笔记五:缩进与选择

    Python最具特色的用缩进来标明成块的代码 缩进 i = 4 j = 2 if i > j: i = i+1 print(i) 这是一个简单的判断,Python的if使用很简单,没有括号等繁琐 ...

  3. HBase调优案例(一)——建表长时间等待最后失败

    现象: 1.在HBase Shell里执行建表操作会等很久,最终失败: 2.通过代码侧进行建表同样不能成功. 原因排查: 1.查询HMaster日志,发现有接收到建表(create)的RPC请求:   ...

  4. phpstudy composer 安装

    今天突然发现phpstudy 可以安装 composer 一打开php中openssl拓展 坑一  我的phpstudy 是2018最新版本,但是你下载laravel什么之类库会报错,是由于compo ...

  5. Mysql 创建函数出现This function has none of DETERMINISTIC, NO SQL, or READS SQL DATA

    This function has none of DETERMINISTIC, NO SQL, or READS SQL DATA in its declaration and binary mys ...

  6. mysql数据库连接错误10060

    今天在使用mysql数据库的时候,出现错误ERROR 2003 (HY000): Can't connect to MySQL server on 'localhost' (10060) 我在网上一顿 ...

  7. 【疑难杂症】Firefox 火狐浏览器 关闭到 http://detectportal.firefox.com 的流量

    日期:2019-07-18 00:02:58 作者:Bay0net 介绍: 0x01. 问题描述 火狐浏览器的时候,抓包会出现很多这样的数据包.  具体的 URL http://detectport ...

  8. wpf datagrid tooltip

    <DataGridTemplateColumn Header="购方名称" Width="260" HeaderStyle="{StaticRe ...

  9. iframe根据子frame的高度自动高度

    <script type="text/javascript"> //光标移动到顶部 this.to_top=function(){ $("html,body& ...

  10. netcore 使用redis session 分布式共享

    首先准备redis服务器(docker 和redis3.0内置的哨兵进行高可用设置) 网站配置Redis作为存储session的介质(配置文件这些略).然后可以了解一下MachineKey这个东西.( ...