1、with语句
再Python中有一些任务,可能事先需要设置,等到任务结束后再进行资源清理的工作。对于这种场景,Python的with语句就提供了一种非常方便的处理方式。一个很好的例子就是文件处理,你需要获取一个文件句柄,从文件中读取数据,然后再关闭文件句柄
使用with语句的话就可以不用再写close语句来关闭文件了,他执行完(离开with语句)后会自动释放资源
在with和as中间的部分就叫做上下文表达式(Context Expression),该表达式要返回一个上下文管理器对象
上下文管理器:一个类对象实现了特殊方法__enter__()和__exit__() (这两种方法会自动执行),则成为这个类对象遵守上下文管理协议,这个类的实例对象被称为上下文管理器
'''
MyConcentMgr类实现了__enter__()和__exit__()方法,成为该类对象遵守了上下文管理协议
该类的实例对象被称作上下文管理器
'''
class MyConcentMgr(object):
def __enter__(self):
print('enter方法被调用执行了')
return self
def __exit__(self, exc_type, exc_val, exc_tb):
print('exit方法被调用执行了')
def show(self):
print('show方法被调用执行了')
'''这是我自定义的一个上下文管理器'''
2、with语句的使用
print(type(open('a.txt','r'))) #open语句自带有__enter__()和__exit__()方法
with open('a.txt','r') as file:
print(file.read())
<class '_io.TextIOWrapper'>
这里的open('a.txt','r')就叫做上下文表达式,这句话所指定的对象就叫做上下文管理器。使用type方法来查看open语句的类型,结果能发现类型是_io.TextIOWrapper,
说明是这个类实现过__enter__()和__exit__()方法,open('a.txt','r')执行所得到的结果就叫做上下文管理器
在创建上下文管理器的同时会创建一个运行时上下文,来自动调用__enter__()方法和__exit__()方法
class MyConcentMgr(object):
def __enter__(self):
print('enter方法被调用执行了')
return self
def __exit__(self, exc_type, exc_val, exc_tb):
print('exit方法被调用执行了')
def show(self):
print('show方法被调用执行了')
with MyConcentMgr() as file: #这里相当于file=MyConcentMgr()
file.show() #在缩进里面的语句统称为with语句体,当他执行完毕后会自动跳出上下文管理器
enter方法被调用执行了
show方法被调用执行了
exit方法被调用执行了
3、with语句的工作原理
(1)执行上下文表达式,并保存起来用于之后的调用
(2)获取上下文管理器的__enter__()方法,如果使用了as子句,则__enter__()方法的返回值赋值给该变量
(3)执行with语句体里面的内容
(4)不论执行过程中是否出现了异常,都会自动执行上下文管理器中的__exit__()方法来释放占用的系统资源
a、如释放资源等。如果执行过程中没有出现异常,或者语句体中执行了语句break/continue/return,则以None作为参数调用__exit__(None, None, None);如果执行过程中出现异常,则使用sys.exc_info得到的异常信息为参数调用__exit__(exc_type, exc_value, exc_traceback);
b、出现异常时,如果__exit__(type, value, traceback)
返回False,则会重新抛出异常,让with
之外的语句逻辑来处理异常,这也是通用做法;如果返回True,则忽略异常,不再对异常进行处理
现在再写文件操作的时候就不用再手动写close语句了,with语句就能自动关闭文件,释放文件资源了
with open('picture.png','rb') as src_file:
with open('copy2picture.png','wb') as target_file:
target_file.write(src_file.read())
4、自定义上下文管理器
class moudle():
def __init__(self):
print('__init__,called')
def __enter__(self):
print('数据已经进来了,马上开始被调用')
return self
def __exit__(self, exc_type, exc_val, exc_tb):
print('程序运行完毕,释放系统资源')
def show(self):
print('测试结果')
return self
with moudle() as mod:
mod.show()
(1条消息) Python中with用法详解_小麦粒的博客-优快云博客https://blog.youkuaiyun.com/u010986753/article/details/96533055(1条消息) python教程:上下文管理器详细教程_Python热爱者的博客-优快云博客_python上下文管理器
https://blog.youkuaiyun.com/qdPython/article/details/124198593