上下文管理器


原理

上下文管理能保证资源会被正确回收,即保证退出步骤的执行。其用处最多的是,作为确保资源被正确回收的一种方式。

一种重复使用的 try-except-finally 结构的有效机制,因为其功能是封闭代码,且可以填充任意格式的代码块。

语法上 __enter__不接受其余参数,除了自带的self之外,__exit__在不包括self参数外,还包括三个参数(一个异常类型,一个异常实例和一个回溯),即exc_typeexc_instancetraceback,默认设置为None

可以通过让一个__exit__方法返回False实现异常传播,或者返回True终止异常。如果抛出另一个异常,将会代码原本的异常被发出去。

编写上下文的场景

  • 资源清理,例如打开数据库,打开文件等需要被正确关闭的资源
  • 避免重复

    1.传播异常

    2.终止异常

    3.处理特定异常类(实例4)

    4.不包括的子类(实例5)

原则1:简单的终止所有异常与try-except语句相同,但终止过多的错误会造成代码调试困难,一些隐含错误也会错误,造成过多的不确性。

代码讲解

实例1

# coding: utf-8

class ContextManager(object):
def __init__(self):
self.entered = False def __enter__(self):
self.entered = True
return self def __exit__(self, exc_type, exc_val, exc_tb):
self.entered = False cm = ContextManager() # 实例化一个对象,如果在其余地方不使用实例对象
# 写成 with ContextManager() as cm:
with cm:
print(cm.entered) # enter 方法返回对象,在 enter 作用范围内 self.enter 为 True print(cm.entered) # 执行了 exit 方法,返回的是 False

实例2

__exit__能处理异常,代码内引起的异常首先传给了__exit__并得到了正常的处理,返回False即异常被重新抛出给主模块,返回True则终止了异常在__exit__方法内

# coding: utf-8

class BubbleExceptions(object):
def __enter__(self):
return self def __exit__(self, exc_type, exc_val, exc_tb):
if exc_val:
print("Bubbing up exception: {}".format(exc_val))
return True with BubbleExceptions():
print(5/0)
"""
__exit__ 返回 False
Bubbing up exception: division by zero
Traceback (most recent call last):
File "/home/yuge/Documents/book/test.py", line 15, in <module>
print(5/0)
ZeroDivisionError: division by zero __exit__ 返回True
Bubbing up exception: division by zero
"""

实例3: 上下文内被正常处理的异常,不会发给__exit__

# coding: utf-8

class BubbleExceptions(object):
def __enter__(self):
return self def __exit__(self, exc_type, exc_val, exc_tb):
if exc_val:
print("Bubbing up exception: {}".format(exc_val))
return False with BubbleExceptions():
try:
print(5/0)
except ZeroDivisionError:
print("这是一个错误!") """
运行结果:
这是一个错误!
"""

实例4:__exit__可以处理特定异常类

即在__exit__方法里封装代码,将处理特定异常类的代码放进去,有点像try-except一样,最多的不同是,其代码可以重用。

# 处理特定的异常,并根据是否提起了其他异常类来决定是否返回 True 或 False
# coding: utf-8 class HandleValueError(object):
def __enter__(self):
return self def __exit__(self, exc_type, exc_val, exc_tb):
if not exc_type:
# 即没有异常出现
return True if issubclass(exc_type, ValueError):
print("捕捉到了一个 ValueError:{}".format(exc_val))
return True return False with HandleValueError():
# raise ValueError("错误的值!")
raise TypeError("错误的类型!") """
raise ValueError("错误的值!")
捕捉到了一个 ValueError:错误的值! raise TypeError("错误的类型!")
Traceback (most recent call last):
File "/home/yuge/Documents/book/test.py", line 22, in <module>
raise TypeError("错误的类型!")
TypeError: 错误的类型! """

实例5:如何处理特定的类,不处理其子类

# coding: utf-8

class ValueErrorSubclass(ValueError):
pass class HandleValueError(object):
def __enter__(self):
return self def __exit__(self, exc_type, exc_val, exc_tb):
if not exc_type:
return True # 这里只处理 ValueError 的错误,即不处理基类的子类
if exc_type == ValueError:
print("处理了一个 ValueError 的错误:{}".format(exc_val))
return True # if issubclass(exc_type, ValueError):
# # 这里处理了包括 基类的 所有类集体
# print("处理了基于 ValueError 的错误:{}".format(exc_val))
# return True return False with HandleValueError():
raise ValueErrorSubclass("这就是个失误") """
包括 issubclass
处理了基于 ValueError 的错误:这就是个失误 注释了 issubclass 之后
Traceback (most recent call last):
File "/home/yuge/Documents/book/test.py", line 30, in <module>
raise ValueErrorSubclass("这就是个失误")
__main__.ValueErrorSubclass: 这就是个失误
"""

python之简述上下文管理的更多相关文章

  1. python with语句上下文管理的两种实现方法

    在编程中会经常碰到这种情况:有一个特殊的语句块,在执行这个语句块之前需要先执行一些准备动作:当语句块执行完成后,需要继续执行一些收尾动作.例如,文件读写后需要关闭,数据库读写完毕需要关闭连接,资源的加 ...

  2. (转)Python中的上下文管理器和Tornado对其的巧妙应用

    原文:https://www.binss.me/blog/the-context-manager-of-python-and-the-applications-in-tornado/ 上下文是什么? ...

  3. Python之面向对象上下文管理协议

    Python之面向对象上下文管理协议 析构函数: import time class Open: def __init__(self,filepath,mode='r',encode='utf-8') ...

  4. Python深入02 上下文管理器

    作者:Vamei 出处:http://www.cnblogs.com/vamei 欢迎转载,也请保留这段声明.谢谢! 上下文管理器(context manager)是Python2.5开始支持的一种语 ...

  5. Python中的上下文管理器和with语句

    Python2.5之后引入了上下文管理器(context manager),算是Python的黑魔法之一,它用于规定某个对象的使用范围.本文是针对于该功能的思考总结. 为什么需要上下文管理器? 首先, ...

  6. 深入理解 Python 中的上下文管理器

    提示:前面的内容较为基础,重点知识在后半段. with 这个关键字,对于每一学习Python的人,都不会陌生. 操作文本对象的时候,几乎所有的人都会让我们要用 with open ,这就是一个上下文管 ...

  7. 【Python】【上下文管理器】

    """#[备注]#1⃣️try :仅当try块中没有异常抛出时才运行else块.#2⃣️for:仅当for循环运行完毕(即for循环没有被break语句终止)才运行els ...

  8. python中实现上下文管理器的两种方法

    上下文管理器: python中实现了__enter__和__exit__方法的对象就可以称之为上下文管理器 实现方法一举例: def File(object): def __init__(self, ...

  9. python 描述符 上下文管理协议 类装饰器 property metaclass

    1.描述符 #!/usr/bin/python env # coding=utf-8 # 数据描述符__get__ __set__ __delete__ ''' 描述符总结 描述符是可以实现大部分py ...

随机推荐

  1. Cocos Code IDE里xcodeprojectlua脚本更新

    lua脚本改动后xcode须要clean又一次编译才干更新,这个是xcode里的老毛病了,网上有一些脚本但不是针对Cocos Code IDE的project文件夹的,这里列出 cocos2dx版本号 ...

  2. emacs 搭建racket开发环境

    emacs 搭建racket开发环境 emacs下搭建开发racket的环境,笔者之前用过下面两种模式:geiser和racket-mode.相对而言,后一种方式要显得简单.本文主要介绍后一种方式环境 ...

  3. 4443: [Scoi2015]小秃玩矩阵|二分答案|匈牙利

    第K大看成第K小各种WA. .. 第K大也就是第n−K+1小.所以就能够愉快的二分答案了 二分答案找出比当前答案小的数的位置的坐标.推断一下能否够选出满足不在同一行同一列的n−K+1个数,然后就能够愉 ...

  4. C++管理指针成员

    1.C++中一般採用以下三种方法之中的一个管理指针成员: (1)指针成员採取常规行为. 这种类具有指针的全部缺陷:具有指针成员且使用默认复制构造函数和赋值操作符,无法避免悬垂指针(两个对象的指针成员指 ...

  5. [WPF]c#调用默认浏览器打开网址

    //调用系统默认的浏览器 System.Diagnostics.Process.Start("http://www.zhaokeli.com");

  6. hdu4849 Wow! Such City!(最短路dijkstra)

    转载请注明出处:http://blog.csdn.net/u012860063? viewmode=contents 题目链接:pid=4849">http://acm.hdu.edu ...

  7. Error解决:Property&#39;s synthesized getter follows Cocoa naming convention for returning &#39;owned&#39;

    在项目中定义了以new开头的textField.结果报错: 先看我的源代码: #import <UIKit/UIKit.h> @interface ResetPasswordViewCon ...

  8. Linux - 硬件杂讲

    先死后活,先记住,再灵活运用. 拍个快照,方便系统坏了,找回. 硬件知识,cpu,内存,i/o总线,电源,机箱. 需求:公司需要做一个内容发布网站,展示公司的信息,你需要选择符合公司要求的Web服务器 ...

  9. MySQL视图、触发器、事务、存储过程、函数

    视图.触发器.事务.存储过程.函数   阅读目录 一 视图 二 触发器 三 事务 四 存储过程 五 函数 六 流程控制 一 视图 视图是一个虚拟表(非真实存在),其本质是[根据SQL语句获取动态的数据 ...

  10. 获取远程请求的IP地址、本机Mac地址和客户端Mac地址

    我在近期项目里面去记录异常日志时,用到了这两个地址,也是从网上和前辈那里学习到的,本人项目是MVC框架的,自己整理了一个公共方法类,包括获取远程客户端IP和Mac地址,以及获取本机Mac地址的方法,代 ...