Python的线程14 有界信号量,这是一个有底线的信号量

正式的Python专栏第51篇,同学站住,别错过这个从0开始的文章!

前篇学委提出了Semaphore信号量来制作限流器的思路和一个简单的限流器实现。

代码运行起来了,看起来没有错误。

不过,细心的读者会发现,如果我们出现像下面的信号量使用方式:

xuewei_semaphore.acquire() #获取信号量
//do something
.... 
xuewei_semaphore.release() #释放信号量
#>>>  xuewei_semaphore.release() #如果这里再调用一次release呢?

也就是 Semaphore 进行的释放的操作,比获取的还多一点点,会不会出问题?

#!/usr/bin/env python
# -*- coding: utf-8 -*-
# @Time : 2021/11/27 10:43 下午
# @Author : LeiXueWei
# @优快云/Juejin/Wechat: 雷学委
# @XueWeiTag: CodingDemo
# @File : thread_semaphore_boundvs_unbound.py
# @Project : hello
import threading
import time

xuewei_semaphore = threading.Semaphore(1) #后面改成有界信号量

print("xuewei_semaphore:", xuewei_semaphore)


def run():
    print(" %s ready" % threading.current_thread().name)
    xuewei_semaphore.acquire()
    print(" %s go" % threading.current_thread().name)
    time.sleep(0.1)
    print(" %s completed" % threading.current_thread().name)
    xuewei_semaphore.release()


def abnormal_run():
    run()
    xuewei_semaphore.release()  # 多release了一次


t = threading.Thread(name="正常使用信号量", target=run)
t.start()
time.sleep(1)
t = threading.Thread(name="非正常使用信号量", target=abnormal_run)
t.start()

上面学委结合了第一篇分享信号量的文章, 简化了一下,重点展示多释放信号量的效果。

我们发现程序没有任何报错:

屏幕快照 2022-01-22 下午11.11.12.png

如果我们把反常的使用增加调用呢,比如下面的代码:

#在上方代码后面加上下面这段代码
for i in range(10000):
    t = threading.Thread(name="非正常使用信号量"+str(i), target=abnormal_run)
    t.start()

全程没有报错!!!

屏幕快照 2022-01-22 下午11.19.34.png

我直接震惊了!Python容错率太高了(这是高情商的说法,😂)

有界信号量 走过来把这个坑填上了

threading.BoundedSemaphore 这个类就是有界信号量,英文名非常直观了。

什么是有界?

跟做人一样,守住底线。那么Semaphore的底线在哪里?我们可以点击这个类去看一部分代码。

Semaphore维护了一个_value, 有界的信号量其实就是加多了一个变量_initial_value,去记录了信号量初始值。

后续,我们调用release函数的时候,有界信号量,每次都会检查,_value是否越界了(就是一旦超过了_initial_value,就抛出一个ValueError,提示越界了!)

这个太友好了.

下面就使用有界信号量展示,对比第一段代码,仅仅把

xuewei_semaphore = threading.Semaphore(1)

改为了:

xuewei_semaphore = threading.BoundedSemaphore(1)

所以读者可以直接改,当然也可以选择直接复制下面的代码再存一个文件运行。

#!/usr/bin/env python
# -*- coding: utf-8 -*-
# @Time : 2021/11/27 10:43 下午
# @Author : LeiXueWei
# @优快云/Juejin/Wechat: 雷学委
# @XueWeiTag: CodingDemo
# @File : thread_semaphore_boundvs_unbound3.py
# @Project : hello
import threading
import time

xuewei_semaphore = threading.BoundedSemaphore(1)

print("xuewei_semaphore:", xuewei_semaphore)


def run():
    print(" %s ready" % threading.current_thread().name)
    xuewei_semaphore.acquire()
    print(" %s go" % threading.current_thread().name)
    time.sleep(0.1)
    print(" %s completed" % threading.current_thread().name)
    xuewei_semaphore.release()


def abnormal_run():
    run()
    xuewei_semaphore.release()  # 多release了一次


t = threading.Thread(name="正常使用信号量", target=run)
t.start()
time.sleep(1)
t = threading.Thread(name="非正常使用信号量", target=abnormal_run)
t.start()

使用有界之后,果然,过度释放,系统就报错了!终于放心了。

屏幕快照 2022-01-22 下午11.22.24.png

总结

大家应该尽量使用有界信号量,它能帮我们守住边界。它不会因为我们不小心写错程序,写多一次release而放任程序继续正常运行。(可能是熬夜写代码,建议不要!写代码需要一个舒适的状态,质量更高)

如果使用纯Semaphore,那请务必封装起来(成对调用封闭起来)不对外开放acquire和release的使用,这样才能安装。

喜欢Python的朋友,请关注学委的 Python基础专栏 or Python入门到精通大专栏

持续学习持续开发,我是雷学委!
编程很有趣,关键是把技术搞透彻讲明白。
欢迎关注微信,点赞支持收藏!

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

雷学委

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值