python with提前退出遇到的坑与解决方案
作者:猪了个去 发布时间:2023-12-24 15:02:35
问题的起源
早些时候使用with实现了一版全局进程锁,希望实现以下效果:
with CacheLock("test_lock", 10):
#如果抢占到锁,那么就执行这段代码
# 否则,让with提前退出
全局进程锁本身不用多说,大部分都依靠外部的缓存来实现的,redis上用的是setnx,有时候根据需要加上缓存击穿问题、随机延后以防止对缓存本身造成压力
当时同样写了单元测试来测试这段代码的有效性:
with CacheLock("test_lock", 10):
value = cache.get("test_lock")
self.assertEqual(value, 1)
with CacheLock("test_lock", 10):
# 不会进到这里
self.assertFalse(True)
value = cache.get("test_lock")
self.assertEqual(value, None)
看起来非常完美地通过了。
这样的一个全局进程锁是通过 __enter__ 方法抛出异常, __exit__ 方法中捕获异常来实现的:
class CacheLock(object):
def __init__(self, lock_key, lock_timeout):
self.lock_key = lock_key
self.lock_timeout = lock_timeout
self.success = False
def __enter__(self):
self.success = cache.lock(self.lock_key, self.lock_timeout)
if self.success:
return self
else:
raise LockException("not have lock")
def __exit__(self, exc_type, exc_value, traceback):
#没有抢到锁的时候,啥都不做?
if self.success:
await cache.delete(self.lock_key)
if isinstance(exc_value, LockException):
return True
if exc_type:
raise exc_value
看起来还不错,毕竟单元测试都过了。
但是,这样的实现是有问题的:
原因在于 __exit__ 的执行不是包在 __enter__ 之外的,因此 __enter__ 抛出的异常,不会被 __exit__ 捕获。
上面的单元测试恰好通过,是因为其中有两个with语句,外面的with 捕获的其实是里面的 __enter__ 抛出的异常
使用改进后的单元测试:
cache.set("test_lock",1)
with CacheLock("test_lock", 10):
self.assertFalse(True)
value = cache.get("test_lock")
self.assertEqual(value, None)
就会发现单元测试过不去了。
这个问题是我试图使用with实现另一个逻辑:AB测试 时出现的,同样是 __enter__ 抛出异常, __exit__ 试图捕获:
import operator
class EarlyExit(Exception):
pass
class ABContext(object):
"""AB测试上下文
>>> with ABContext(newVersion, consts.ABEnum.layer2):
>>> # dosomething
"""
def __init__(self, version, ab_layer, relationship="eq"):
self.version = version
self.ab_layer = ab_layer
# 如果不存在这种操作符,那就提前报错
self.relationship = getattr(operator, relationship)
def __enter__(self):
# 如果不满足条件,等于不执行上下文中的内容
if not self.relationship(self.version, self.ab_layer.value):
raise EarlyExit("not match")
return self
def __exit__(self, exc_type, exc_value, traceback):
if exc_value is None:
return True
if isinstance(exc_value, EarlyExit):
return True
if exc_type:
raise exc_value
return True
调试没有通过的单元测试的时候发现,抛出异常后根本没有执行到 __enter__
第一种解决方案
既然想明白了with的执行顺序,那么第一种解决方案就呼之欲出了:既然__exit__捕获的异常在__enter__执行完成之后,那么我们提供一个函数确认一下就可以了,把ABContext实现改成这样:
def ensure(self):
if not self.relationship(self.version, self.ab_layer.value):
raise EarlyExit("not match")
def __enter__(self):
# 如果不满足条件,等于不执行上下文中的内容
return self
使用的时候:
with ABContext(newVersion, consts.ABEnum.layer2) as c:
c.ensure()
# 执行其他的想要执行的代码
但这样的解决方法并不优雅,万一使用这个ABContext的时候忘记用ensure方法了,那么就等于完全没用这个Context方法,太容易失误了,而且代码也失去了Pythonic的性质
第二种解决方法
翻了一下contextlib的标准库文档,发现有一个已经废弃的函数: contextlib.nested
from contextlib import nested
with nested(*managers):
do_something()
可以执行多个上下文.
from contextlib import nested
with nested(A(), B(), C()) as (X, Y, Z):
do_something()
# is equivalent to this:
m1, m2, m3 = A(), B(), C()
with m1 as X:
with m2 as Y:
with m3 as Z:
do_something()
这个废弃的特性在Python2.7之后,可以直接由with关键字执行,形如:
with context1,context2:
#do something
这个特性还不错,根据 __enter__ 的执行顺序的话,那么我们可以实现一个由第一个 context的__exit__来捕获,第二个context的__enter__来抛出异常,
如同这样:
class AlwaySuccessContext(object):
def __enter__(self):
return self
def __exit__(self, exc_type, exc_value, traceback):
if isinstance(exc_value, EarlyExit):
return True
if exc_type:
raise exc_value
return True
结合前面我们实现的ABContext的使用是这样的:
def test_context_noteq(self):
obj = MagicMock(return_value=True)
with AlwaySuccessContext(), ABContext(2, const.ABTestEnum.control):
self.assertFalse(obj())
obj.assert_not_called()
good,单元测试就这样过了
能不能再给力点?
确实,在with里要写俩context有点蛋疼,并不是特别优雅,能不能还是回到最初的那种用法:我们只用写一条context,这一个context做到了两个context的事情?
要是nested那个函数还在就好了。。要的其实就是它的功能。
Python3.1之后contextlib提供了一个ExitStack的功能来提供一个模拟的功能,但试了一下发现,实际上只调用了__enter__方法,但没有做对应的异常捕获
第三种解决方案
哈哈哈哈把自己绕到圈子里去了,想了一下,同样是一个缩进的代码块,为什么不能用if来解决呢!不就是个:
def test_context_noteq(self):
# 不等的时候,不会执行with里的内容
obj = MagicMock(return_value=True)
context = ABContext(2, const.ABTestEnum.control)
# print(type(context))
if ABContext(2, const.ABTestEnum.control):
self.assertFalse(obj())
obj.assert_not_called()
TIL
总之学到了contextlib里的一些有用的函数和装饰器,也第一次发现with可以放个context
虽然放多个context的动态构造还有待研究,with 后面的代码块也不能填一个元组或者列表。。惆怅。。
来源:https://zhuanlan.zhihu.com/p/32617838?utm_source=tuicool&utm_medium=referral


猜你喜欢
- 本文实例为大家分享了Vue实现导航栏菜单的具体代码,供大家参考,具体内容如下这里是刚学习vue的时候,没有用vue的任何UI组件库写的导航栏
- Eric A. Meyer 对基于 Web 标准的 CSS 与 HTML 绝非一知半解,他是这个领域杰出的专家,曾写过不少 CSS 方面的书
- 1.5 学习ASP.net 的过程中如何求助--加入 ASPNG 讨论列表 Charles Carroll 作为不断壮大的 ASP.NET
- Mysql的connector/net5.0下载地址: http://dev.mysql.com/get/Downloads/Connect
- 每种语言都有自己的优势,互相结合起来各取所长程序执行起来效率更高或者说哪种实现方式较简单就用哪个,nodejs是利用子进程来调用系统命令或者
- 方法一:<span style="font-size:14px;">#read txt method one
- event.keycode值大全 1 keycode 8 = BackSpace BackSpace 2 keycode 9 = Tab T
- 前言Flask是一个使用python编写的轻量级Web框架,对比其他相同类型的框架而言,这个框架更加的灵活轻便。并且具有很强的定制性,用户可
- 本文介绍了Vue2 SSR 缓存 Api 数据,分享给大家,具体如下:1. 安装缓存依赖: lru-cachenpm install lru
- Google以其简洁的搜索框引领着互联网,搜索系统似乎成了每个网站必备品,甚至于是那些本身几乎是由静态页面组成的企业网站都要来个搜索功能,这
- 在开发中我们经常遇到这样的需求,需要用户直接点击一个链接进入到一个页面,用户点击后链接后会触发401拦截返回登录界面,登录后又跳转到链接的页
- 直方图,又称质量分布图,是一种统计报告图,由一系列高度不等的纵条或线段表示数据分布情况。用横轴表示数据类型,纵轴表示分布情况。直方图是数值数
- 对于小型站点,使用七牛云存储的免费配额已足够为站点提供稳定、快速的存储服务七牛云存储已有Python SDK,对它进行简单封装后,就可以直接
- 视频读取视频读取,主要利用VideoCapture类下的方法打开视频并获取视频中的帧,具体示例如下:#include<iostream
- 在该网站下载你所需要的mysql依赖驱动版本,比如我的QT版本是5.15就下5.15版的 按照你的Qt编译版本和机器位数进行下载,
- 目录一、前言二、方法1、代码2、运行一、前言SpringBoot作为后端开发框架,有强大且方便的处理能力。但是作为一个结合数据分析+前台展示
- 1.文件结构MySQLdb和pymysql的使用差不多阅读的小伙伴可以自己尝试实现2.实验效果3.主文件:main.pyimport MyS
- 本文实例讲述了Python实现向服务器请求压缩数据及解压缩数据的方法。分享给大家供大家参考,具体如下:向服务器请求压缩数据格式,并解压缩数据
- Python是一个很酷的语言,因为你可以在很短的时间内利用很少的代码做很多事情。不仅如此,它还能轻松地支持多任务,比如多进程等。Python
- 通过学习私有函数与私有变量,可以更好的完善 类的开发 ,从而丰满我们的场景与实现方案。什么是私有函数和私有变量私有函数与私有变量中的私有是什