我有一个类,其对象使用需要清理的系统资源(例如,创建临时文件).至少,这需要在程序结束时发生.到目前为止,例如,我一直在使用contextmanager来执行此操作
@contextlib.contextmanager
def tempdir(prefix='tmp'):
tmpdir = tempfile.mkdtemp(prefix=prefix)
try:
yield tmpdir
finally:
shutil.rmtree(tmpdir)
with tempdir() as tmp:
do_something(tmp)
这样可以确保在我使用完资源后立即将其释放,并且对于我的大多数目的而言,这很好用.但是,我发现当交互式使用python时(例如在ipython中),此模型会崩溃.在这种情况下,整个会话将需要放在with块内,这不是很实际(实际上,ipython只会一次评估所有块,因此整个交互性都会消失).
有没有办法确保在清理资源的同时仍可在交互式会话中使用?在C语言中,可以通过将清理放入析构函数中来实现. python __del__语句类似,但是有人告诉我__del__不可靠,应避免使用.特别是it is not guaranteed that __del__
will be called for objects that still exist when the interpreter exits.这是否是__del__仍然是最佳解决方案的情况之一?并且,如果contextmanager方法与交互式会话不兼容,为什么建议将其作为最“ pythonic”的方法?
解决方法:
我认为您的问题没有灵丹妙药,但是您可以通过以下方式解决问题:
>显式调用处理资源的方法(您的close方法).唯一的缺点是明确性.
>为交互式解释器创建一个薄包装器,该薄包装器使用atexit
module在退出时注册实例的close方法.缺点是您的所有资源通常将在您想要的之后释放.
>创建辅助函数(动态创建它们并不困难),该函数包装了资源的使用情况.如果您需要使用一种资源调用更多函数,这是不可行的.例如:
def do_something()
with tempdir() as tmp:
return original.do_something(tmp)
>创建一个隐藏资源处理的解决方案.例如,我不在乎TCP套接字,ssl,301/302重定向,打开证书文件等,我只需要使用一个特定的证书通过https发送GET请求.
自然,这取决于您要解决的问题.