第15章 上下文管理器和else块
#《流流畅的Python》第15章 上下文管理器和else块
#15.1 先做这个,再做那个:if语句之外的else块
#else子句不仅能在if语句中使用,还能在for、while和try语句中使用。
#for/else、while/else和try/else的语义关系紧密,不过与if/else差别很大。
#可是,在循环中,else的语义恰好相反:“运行这个循环,然后做那件事。
#在Python中,try/except不仅用于处理错误,还常用于控制流程。
#EAFP取得原谅比获得许可容易(easier to ask for forgiveness than permission)。
#15.2 上下文管理器和with块
#上下文管理器协议包含 __enter__ 和 __exit__ 两个方法。
#with语句开始运行时,会在上下文管理器对象上调用 __enter__ 方法。
#with语句运行结束后,会在上下文管理器对象上调用 __exit__ 方法,以此扮演finally子句的角色。
with open('mirror.py')as fp: #fp绑定到打开的文件上,因为文件的__enter__方法返回self。
src=fp.read(60) #从fp中读取一些数据。
print(len(src))
print(fp) #fp变量仍然可用。
print(fp.closed,fp.encoding) #可以读取fp对象的属性。
fp.read()#但是不能在fp上执行I/O操作,因为在with块的末尾,调用TextIOWrapper.__exit__方法把文件关闭了。
#示例15-2 测试LookingGlass上下文管理器类
class LookingClass:
def __enter__(self):
import sys
self.original_write=sys.stdout.write
sys.stdout.write=self.reverse_write
return 'JABBERWOCKY'
def reverse_write(self,text):
self.original_write(text[::-1])
def __exit__(self, exc_type, exc_value, exc_traceback):
# 如果一切正常Python调用__exit__方法时传入的参数是None,None, None;如果抛出了异常,这三个参数是异常数据,
import sys
sys.stdout.write=self.original_write
if exc_traceback is ZeroDivisionError:
print('Please DO NOT divide by zero')
return True
with LookingClass() as what:
print("Alice,kitty and Snowdrop") #pordwonS dna yttik,ecilA
print(what) #YKCOWREBBAJ
print(what) #JABBERWOCKY
print('Back to normal') #Back to normal
#示例15-4 在with块之外使用LookingGlass类
manager=LookingClass()
print(manager) #<__main__.LookingClass object at 0x004B5AB0>
monster=manager.__enter__()
print(monster=='JABBERWOCKY') #eurT
print(monster) #YKCOWREBBAJ
print(manager) #>0BA50400x0 ta tcejbo ssalCgnikooL.__niam__<
print(monster) #JABBERWOCKY'
#15.3 contextlib模块中的实用工具
#closing如果对象提供了close()方法,但没有实现__enter__/__exit__协议,那么可以使用这个函数构建上下文管理器。
#suppress构建临时忽略指定异常的上下文管理器。
#@contextmanager这个装饰器把简单的生成器函数变成上下文管理器,这样就不用创建类去实现管理器协议了。
#ContextDecorator这是个基类,用于定义基于类的上下文管理器。这种上下文管理器也能用于装饰函数,在受管理的上下文中运行整个函数。
#ExitStack这个上下文管理器能进入多个上下文管理器。with 块结束时,ExitStack 按照后进先出的顺序调用栈中各个上下文管理器的__exit__ 方法。
#如果事先不知道 with 块要进入多少个上下文管理器,可以使用这个类。例如,同时打开任意一个文件列表中的所有文件。
#15.4 使用@contextmanager
#在使用 @contextmanager 装饰的生成器中,yield语句的作用是把函数的定义体分成两部分:
#yield语句前面的所有代码在with块开始时(即解释器调用 __enter__ 方法时)执行,yield语句后面的代码在with 块结束时(即调用 __exit__ 方法时)执行。
import contextlib
@contextlib.contextmanager
def looking_glass():
import sys
original_write=sys.stdout.write
def reverse_write(text):
original_write(text[::-1])
sys.stdout.write=reverse_write
yield 'JABBERWOCKY'
sys.stdout.write=original_write
with looking_glass() as what:
print('Alice,kitty and Snowdrop') #pordwonS dna yttik,ecilA
print(what) #YKCOWREBBAJ
print(what) #JABBERWOCKY
#示例 15-8用于原地重写文件的上下文管理器
import csv
with inplace(csvfilename,'r',newline='') as(infh,outfh):
reader=csv.reader(infh)
writer=csv.writer(outfh)
for row in reader:
row += ['new','columns']
writer.writerow(row)
#15.5 本章小结
#除了自动关闭打开的文件之外,with语句还有很多用途。
#@contextmanager装饰器优雅且实用,把三个不同的Python特性结合到了一起:函数装饰器、生成器和with语句。
第15章 上下文管理器和else块的更多相关文章
- 流畅的python第十五章上下文管理器和else块学习记录
with 语句和上下文管理器for.while 和 try 语句的 else 子句 with 语句会设置一个临时的上下文,交给上下文管理器对象控制,并且负责清理上下文.这么做能避免错误并减少样板代码, ...
- Python 上下文管理器和else块
最终,上下文管理器可能几乎与子程序(subroutine)本身一样重要.目前,我们只了解了上下文管理器的皮毛--Basic 语言有with 语句,而且很多语言都有.但是,在各种语言中 with 语句的 ...
- 上下文管理器和else块
一.if 语句之外的 else块 else 子句不仅能在 if 语句中使用,还能在for.while和try语句中使用. (1)for :仅当 for 循环运行完毕时(即 for 循环没有被break ...
- 『流畅的Python』第15章:上下文管理器和else块
- 【Python】【上下文管理器】
"""#[备注]#1⃣️try :仅当try块中没有异常抛出时才运行else块.#2⃣️for:仅当for循环运行完毕(即for循环没有被break语句终止)才运行els ...
- 流畅python学习笔记:第十五章:上下文管理器
在开始本章之前,我们首先来谈谈try-excep..final模块.在Python中,进行异常保护的最多就是用try..except..final.首先来看下下面的代码.进行一个简单的除法运算.为了防 ...
- python上下文管理器ContextLib及with语句
http://blog.csdn.net/pipisorry/article/details/50444736 with语句 with语句是从 Python 2.5 开始引入的一种与异常处理相关的功能 ...
- python之with语句结合上下文管理器
所谓上下文管理器即在一个类中重写了__enter__方法和__exit__方法的类就可以成为上下文管理器类. 我们可以通过with语句结合上下文管理器简化一些操作. 使用with语句结合自定义上下文管 ...
- Python核心技术与实战——二一|巧用上下文管理器和with语句精简代码
我们在Python中对于with的语句应该是不陌生的,特别是在文件的输入输出操作中,那在具体的使用过程中,是有什么引伸的含义呢?与之密切相关的上下文管理器(context manager)又是什么呢? ...
随机推荐
- sizeof 4字节对齐
#include <iostream> #include<assert.h> using namespace std; typedef struct sys{ char a; ...
- selenium验证码和错误截图
验证码的识别: 1,破解验证码 OCR识别(一般使用tesseract-ocr) 人工智能(AI机器学习 TensorFlow,成本大) 2,绕过验证码 1, 让开发人员临时关闭验证码 2,提供万能验 ...
- 通过docker-compose构建ghost博客(二)
上一篇通过yml文件构建 ghost博客,这次通过构建nginx服务,并添加反向代理来运行搭建的ghost博客. 目录结构 ghost.conf 就是 定义的nginx 加载的配置文件 server ...
- 针对appium的webdriver执行swipe无效的解决办法
self.driver.swipe(x1,y1,x2,y1,t) 当时代码里有如上这么一句,当时源码是这么说的: # convenience method added to Appium (NOT S ...
- Excel中的常用功能
Excel 是微软办公套装软件的一个重要的组成部分,它可以进行各种数据的处理.统计分析和辅助决策操作,广泛地应用于管理.统计财经.金融等众多领域. Excel删除公式保留数据 1.简单举例一个简单的公 ...
- Django模板如何用一个变量查找字典值
mydict = {"key1":"value1", "key2":"value2"} 在Django模板中查找字典值的 ...
- 整理 oracle异常错误处理
5.1 异常处理概念 5.1.1 预定义的异常处理 5.1.2 非预定义的异常处理 5.1.3 用户自定义的异常处理 5.1.4 用户定义的异常处理 5.2 异常错误传播 5.2.1 在执行部分引发 ...
- 5I - 汉诺塔IV
还记得汉诺塔III吗?他的规则是这样的:不允许直接从最左(右)边移到最右(左)边(每次移动一定是移到中间杆或从中间移出),也不允许大盘放到小盘的上面.xhd在想如果我们允许最大的盘子放到最上面会怎么样 ...
- hdu 5491(2015合肥网赛)The Next
题目;http://acm.hdu.edu.cn/showproblem.php?pid=5491 题意就是,T组测试数据.然后L,S1,S2.L的二进制中有x个1,x满足 S1<=x< ...
- Find Amir CodeForces 805C
http://codeforces.com/contest/805/problem/C 题意:有n个学校,学校的编号是从1到n,从学校i到学校j的花费是(i+j)%(n+1),让你求遍历完所有学校的最 ...