Python上下文管理器的高级使用

在文件处理和网络编程时,对于打开的文件不管最后内容处理是否符合预期都要在结束时关闭文件。这时常见的处理方法是try catch finally 的方法
f = open("demo.txt", "a")
try:
f = open("demo.txt", "a")
raise Exception("读取内容报错")
except Exception as e:
print(e)
finally:
if f is not None:
f.close()
使用错误捕获的方法有效的避免了文件打开没有关闭的情况。实现同样的功能有一种更加优雅的方法,那就是with关键字。
with 语法介绍:
上下文管理器(context manager)是 Python2.5 开始支持的一种语法,用于规定某个对象的使用范围。一旦进入或者离开该使用范围,会有特殊操作被调用。它的语法形式是 with…as…,主要应用场景资源的创建和释放。例如,文件就支持上下文管理器,可以确保完成文件读写后关闭文件句柄。
简单示例如下:
with open("demo.txt", "a+") as f:
print("文件读取")
使用with就可以不用写冗长的try catch finally等处理流程,而且文件一定是安全打开和关闭。with之所以能做到是因为open对象中有打开文件的方法__enter__和关闭文件的方法__exit__,在执行到print之前with调用了__enter__方法,执行完print之后with调用了__exit__方法。并且不管打开文件之后是否会出错,with最终都会调用__eixt__方法。所以文件一定能安全关闭。
with 使用
一个对象想要支持with这种优雅的方法来管理,需要实现__enter__和__exit__两个方法。根据with是否有返回,可以分成两个类型。
- 没有返回值,如
with open("demo.txt"); - 有返回值,如
with open("demo.txt") as f;
没有返回值
class ContextDemo:
def __init__(self):
print("__init__")
return
def __enter__(self):
print("__enter__")
def __exit__(self, exc_type, exc_val, exc_tb):
print("__exit__")
if __name__ == '__main__':
with ContextDemo():
print("我就是with中的操作")
__init__
__enter__
我就是with中的操作
__exit__
有返回值
with 的返回值实际上就是__enter__中return的结果。有return就有返回值,没有return就没有返回值
class ContextDemo:
def __init__(self):
print("__init__")
return
def __enter__(self):
print("__enter__")
return 100
def __exit__(self, exc_type, exc_val, exc_tb):
print("__exit__")
if __name__ == '__main__':
with ContextDemo() as num :
print(f"我就是with中的操作,输出{num}")
__init__
__enter__
我就是with中的操作,输出100
__exit__
contextlib
contextlib是一个上下文管理器工具,工具中有一个contextmanager,可以将一个函数变成一个装饰器,用来支持with关键字。
通俗来说就是如果一个函数想要支持with就需要有__enter__和__exit__这两个方法。而contextmanager就可以将函数变成这样一个函数。
依然根据with是否有返回值,区分两中使用方法
不带返回值
from contextlib import contextmanager
@contextmanager
def manager_fun():
print("enter")
yield
print("exit")
with manager_fun():
print("我就是with中的操作")
enter
我就是with中的操作
exit
函数中通过yield将函数分成三个部分:yield之前代码、yield之后代码、yield;
执行的流程也是三个步骤:
with manager_funwith 进入时执行yield之前的代码部分print("enter")- 遇到yield中断,然后执行
print("我就是with中的操作") - 最后返回到yield后面,执行
print("exit")
传统上下文管理器需要实现的__enter__和__exit__在contextlib中对应的就是:yield之前代码就是__enter__,yield代码之后就是__exit__
带返回值
from contextlib import contextmanager
@contextmanager
def manager_fun():
print("enter")
yield 100
print("exit")
with manager_fun() as num:
print(f"我就是with中的操作{num}")
enter
我就是with中的操作100
exit
with 可以有返回值,如果想实现返回值,在传统上下文管理器中是__enter__函数的返回值就是with的返回值,在contextlib中就是yield的返回值。
yield 返回一个值,然后程序中断暂停在这里。返回的值就是with的返回值。(听起来有点绕口)
contextlib 使用场景
有些场景下想要使用with来管理对象,但是目标对象并不支持,这时就可以通过contextlib来包装一个上下文管理器,达到管理资源的目的
Python上下文管理器的高级使用的更多相关文章
- Python上下文管理器
在Python中让自己创建的函数.类.对象支持with语句,就实现了上线文管理协议.我们经常使用with open(file, "a+") as f:这样的语句,无需手动调用f.c ...
- python 上下文管理器
作者:Vamei 出处:http://www.cnblogs.com/vamei 欢迎转载,也请保留这段声明.谢谢! 上下文管理器(context manager)是Python2.5开始支持的一种语 ...
- Python 上下文管理器和else块
最终,上下文管理器可能几乎与子程序(subroutine)本身一样重要.目前,我们只了解了上下文管理器的皮毛--Basic 语言有with 语句,而且很多语言都有.但是,在各种语言中 with 语句的 ...
- python上下文管理器ContextLib及with语句
http://blog.csdn.net/pipisorry/article/details/50444736 with语句 with语句是从 Python 2.5 开始引入的一种与异常处理相关的功能 ...
- Python上下文管理器 with
对于系统资源的操作,如:文件操作.数据库操作等,我们往往打开文件.连接数据库后忘了将其close掉,这时就可能会引发异常,因此我们常用的做法是: # coding:utf-8 f = open(&qu ...
- Python上下文管理器(Context managers)
上下文管理器(Context managers) 上下文管理器允许你在有需要的时候,精确地分配和释放资源. 使用上下文管理器最广泛的案例就是with语句了.想象下你有两个需要结对执行的相关操作,然后还 ...
- python上下文管理器细读
test 1 上下文管理器,将生成器转化为上下文管理器 import contextlib @contextlib.contextmanager def a(): print(1) yield pri ...
- 吃透Python上下文管理器
什么是上下文管理器? 我们常见的with open操作文件,就是一个上下文管理器.如: with open(file, 'rb') as f: text = f.read() 那上下文管理器具体的定义 ...
- python 上下文管理器contextlib.ContextManager
1 模块简介 在数年前,Python 2.5 加入了一个非常特殊的关键字,就是with.with语句允许开发者创建上下文管理器.什么是上下文管理器?上下文管理器就是允许你可以自动地开始和结束一些事情. ...
- Python 上下文管理器模块--contextlib
在 Python 处理文件的时候我们使用 with 关键词来进行文件的资源关闭,但是并不是只有文件操作才能使用 with 语句.今天就让我们一起学习 Python 中的上下文管理 contextlib ...
随机推荐
- include 0。0
参考好文 php://filter的各种过滤器_php://filter过滤器种类-CSDN博客 打开页面是一段php代码 可以知道flag在flag.php文件里面,然后执行没有结果,就只能用文件读 ...
- Netty源码学习7——netty是如何发送数据的
零丶引入 系列文章目录和关于我 经过<Netty源码学习4--服务端是处理新连接的&netty的reactor模式和<Netty源码学习5--服务端是如何读取数据的>,我们了 ...
- JOISC2017 题解
\(\text{By DaiRuiChen 007}\) Contest Link A. Cultivation Problem Link 题目大意 在一个 \(r\times c\) 的网格上有 \ ...
- 编辑linux服务启动命令(app-script.sh命令编写)
#!/bin/sh# 注:这里可替换为你自己的执行程序,其他代码无需更改APP_NAME=app-biz.jar #使用说明,用来提示输入参数usage() { echo "Usage: s ...
- [ARC122E] Increasing LCMs
Problem Statement We have a sequence of $N$ positive integers: $A_1,A_2,\cdots,A_N$. You are to rear ...
- [ABC317G] Rearranging
Problem Statement There is a grid with $N$ rows and $M$ columns. The square at the $i$-th row from t ...
- Net 高级调试之十三:托管堆的几个经典破坏问题
一.介绍 今天是<Net 高级调试>的第十三篇文章,这篇文章写作时间的跨度有点长.这篇文章我们主要介绍 经典的案例,如何查找问题,如何解决问题,最重要我们看到了问题,要有解决的思路,没有思 ...
- Cocos-JS HTTP网络请求
网络结构 网络结构是网络构建方式,目前流行的有客户端服务器结构(C/S结构)和点对点(P2P)结构网络. 客户端服务器结构(C/S结构) 这种结构又被称为Clicent/Server结构,它是一种主从 ...
- python、numpy计算不同文档下的词的TF-IDF值以及进行文档相似度匹配(实战)
TF-IDF 计算公式(一个词的 tf-idf 值在不同文档,它的值也不同): 1.根据已有的原始数据,只展示了前5片文档,content是文档内容,s_words是通过jieba分词将文档划分成了若 ...
- python tkinter 使用(七)
python tkinter 使用(七) 本篇文章主要讲下tkinter 中的message 控件. Message控件可以用于在窗口中显示一段文本消息. 以下是个简单的例子: #!/usr/bin/ ...