python with语句的原理与用法详解


Posted in Python onMarch 30, 2020

本文实例讲述了python with语句的原理与用法。分享给大家供大家参考,具体如下:


之前看到一篇博客说博主python面试时遇到面试官提问with的原理,而那位博主的博文没有提及with原理,故有此文。

关于with语句,官方文档中是这样描述的:

The with statement is used to wrap the execution of a block with methods defined by a context manager (see section With Statement Context Managers). This allows common try...except...finally usage patterns to be encapsulated for convenient reuse.

with_stmt ::= "with" with_item ("," with_item)* ":" suite

with_item ::= expression ["as" target]

The execution of the with statement with one “item” proceeds as follows:

The context expression (the expression given in the with_item) is evaluated to obtain a context manager.

The context manager's __exit__() is loaded for later use.

The context manager's __enter__() method is invoked.

If a target was included in the with statement, the return value from __enter__() is assigned to it.

Note

The with statement guarantees that if the __enter__() method returns without an error, then __exit__() will always be called. Thus, if an error occurs during the assignment to the target list, it will be treated the same as an error occurring within the suite would be. See step 6 below.

The suite is executed.

The context manager's __exit__() method is invoked. If an exception caused the suite to be exited, its type, value, and traceback are passed as arguments to __exit__(). Otherwise, three None arguments are supplied.

谷歌翻译成中文就是:

with语句用于使用由上下文管理器定义的方法来封装块的执行(请参见使用语句上下文管理器一节)。 这允许通用的try…except…finally使用模式被封装以便于重用【这句话大概意思就是“with语句”类似于try…except…finally封装之后的的情况】。

带有一个“项目”的with语句的执行过程如下:
1.上下文表达式(在with_item中给出的表达式)被评估以获得上下文管理器。【会区分类型来处理,如文件,进程等都可以使用with语句】
2.上下文管理器的__exit __()被加载供以后使用。【负责上下文的退出】
3.上下文管理器的__enter __()方法被调用。【负责上下文的进入】
4.如果在with语句中包含目标,则将__enter __()的返回值分配给它。【如果with后面跟着as 对象(如with open() as f),那么此对象获得with上下文对象的__enter__()的返回值,(附:应该是类似操作数据库时的连接对象和游标的区别)】

注意
with语句保证,如果__enter __()方法返回时没有错误,那么将始终调用__exit __()。 因此,如果在分配给目标列表期间发生错误,它将被视为与套件内发生的错误相同。 请参阅下面的第6步。

5.该套件已执行。【意思就是语句体中的过程执行完毕,执行完毕就到第六步--调用__exit__()来退出】
6.上下文管理器的__exit __()方法被调用。 如果异常导致套件退出,则其类型,值和回溯作为参数传递给__exit __()。 否则,将提供三个无参数。

关于退出返回值:

If the suite was exited due to an exception, and the return value from the __exit__() method was false, the exception is reraised. If the return value was true, the exception is suppressed, and execution continues with the statement following the with statement.

If the suite was exited for any reason other than an exception, the return value from __exit__() is ignored, and execution proceeds at the normal location for the kind of exit that was taken.

中文:
如果套件由于异常而退出,并且__exit __()方法的返回值为false,则会重新对异常进行重新评估。 如果返回值为true,则异常被抑制,并继续执行with语句后面的语句。

如果套件由于除了异常之外的任何原因而退出,则__exit __()的返回值将被忽略,并且执行将在正常位置继续进行。

意思就是:

如果是异常退出,那么会返回false,(根据文档中的exit的描述“that __exit__() methods should not reraise the passed-in exception; this is the caller's responsibility.”,大概意思就是exit()不会处理异常,会重新抛出异常抛出给外面,由调用者处理,因为这是调用者的责任)

如果返回 True,则忽略异常,不再对异常进行处理【(在exit内部处理完异常后,可以让”__exit__()”方法返回True,此时该异常就会不会再被抛出,with会认为它的执行体没有发生异常)】

(with会识别返回值,根据返回值来处理,如果是False,那么with会将执行体中的异常抛出,如果是True,那么with会认为没有发生异常(忽略异常),而继续执行外面的语句,但由于内部调用的了__exit__(),所以在异常之后的语句是不会运行的

附上一个文档中提供的一个关于with中使用锁的例子:

python with语句的原理与用法详解

几个测试:

1.执行体中发生异常:

import time
class myContextDemo(object):
 def __init__(self,gen):
  self.gen = gen
 def __enter__(self):
  print("enter in ")
  return self.gen
 def __exit__(self, exc_type, exc_val, exc_tb):
#exc_type是exception_type exc_val是exception_value exc_tb是exception_trackback
  print("exit in ")
  if exc_type is None:#如果是None 则继续执行
   print("None:",exc_type, exc_val, exc_tb)

  else: #异常不为空时执行,这一步,如果with语句体中发生异常,那么也会执行
   print("exception:", exc_type, exc_val, exc_tb)
   print("all done")

if __name__=="__main__":
 gen=(i for i in range(5,10))
 G=myContextDemo(gen)
 with G as f :
  print("hello")
  for i in f:
   print(i,end="\t")
  #测试1:执行体中发生异常
  raise Exception("母鸡啊")
 print("main continue")

结果显示:python with语句的原理与用法详解

1.抛出异常后,后面main continue不再执行

2.__exit__()中的else会执行

测试2:当else中强制返回为True时:

import time
class myContextDemo(object):
 def __init__(self,gen):
  self.gen = gen
 def __enter__(self):
  print("enter in ")
  return self.gen
 def __exit__(self, exc_type, exc_val, exc_tb):
#exc_type是exception_type exc_val是exception_value exc_tb是exception_trackback
  print("exit in ")
  if exc_type is None:#如果是None 则继续执行
   print("None:",exc_type, exc_val, exc_tb)

  else: #异常不为空时执行,这一步,如果with语句体中发生异常,那么也会执行
   print("exception:", exc_type, exc_val, exc_tb)
   print("all done")
   return True #这里如果返回true可以看到发生异常后,main continue可以执行
   #即,如果exc_type是true,那么会继续执行,实际上,也可以在这里处理一下异常再返回true


if __name__=="__main__":
 gen=(i for i in range(5,10))
 G=myContextDemo(gen)
 with G as f :
  print("hello")
  for i in f:
   print(i,end="\t")
  raise Exception("母鸡啊")
  # print("continue")#这里不会执行
 print("main continue")

结果显示:python with语句的原理与用法详解

1.返回True之后,with会忽略异常,继续执行,所以这里“main continue”能执行

2.即使忽略异常,在with体中异常之后的语句依旧不会执行

附:理论上可以在返回True之前处理一下异常

PS:如果大家想要了解得更详细,可以自己尝试去读一下官方文档。

附上关于with语句的详细介绍官方文档:https://www.python.org/dev/peps/pep-0343/

希望本文所述对大家Python程序设计有所帮助。

Python 相关文章推荐
Python实现批量将word转html并将html内容发布至网站的方法
Jul 14 Python
python与C互相调用的方法详解
Jul 14 Python
MAC中PyCharm设置python3解释器
Dec 15 Python
ubuntu安装mysql pycharm sublime
Feb 20 Python
浅谈Python中的作用域规则和闭包
Mar 20 Python
Python实现操纵控制windows注册表的方法分析
May 24 Python
基于pytorch的保存和加载模型参数的方法
Aug 17 Python
Python列表list常用内建函数实例小结
Oct 22 Python
new_zeros() pytorch版本的转换方式
Feb 18 Python
使用python实现名片管理系统
Jun 18 Python
linux系统下pip升级报错的解决方法
Jan 31 Python
python中使用asyncio实现异步IO实例分析
Feb 26 Python
对django 2.x版本中models.ForeignKey()外键说明介绍
Mar 30 #Python
Python进程的通信Queue、Pipe实例分析
Mar 30 #Python
基于Django OneToOneField和ForeignKey的区别详解
Mar 30 #Python
django 扩展user用户字段inlines方式
Mar 30 #Python
Python3标准库之threading进程中管理并发操作方法
Mar 30 #Python
解决django xadmin主题不显示和只显示bootstrap2的问题
Mar 30 #Python
Python2 与Python3的版本区别实例分析
Mar 30 #Python
You might like
YII2.0之Activeform表单组件用法实例
2016/01/09 PHP
PHP实现数组转JSon和JSon转数组的方法示例
2018/06/14 PHP
php代码调试利器firephp安装与使用方法分析
2018/08/21 PHP
PHP接入微信H5支付的方法示例
2019/10/28 PHP
javascript游戏开发之《三国志曹操传》零部件开发(三)情景对话中仿打字机输出文字
2013/01/23 Javascript
javascript中的void运算符语法及使用介绍
2013/03/10 Javascript
Jquery attr("checked") 返回checked或undefined 获取选中失效
2013/10/10 Javascript
JS格式化数字金额用逗号隔开保留两位小数
2013/10/18 Javascript
解决js函数闭包内存泄露问题的办法
2016/01/25 Javascript
关于vue.js发布后路径引用的问题解决
2017/08/15 Javascript
微信小程序实现无限滚动列表
2020/05/29 Javascript
JavaScript设计模式之代理模式实例分析
2019/01/16 Javascript
小程序实现列表多个批量倒计时
2021/01/29 Javascript
Vue中computed、methods与watch的区别总结
2019/04/10 Javascript
如何根据业务封装自己的功能组件
2019/04/19 Javascript
在Vue项目中用fullcalendar制作日程表的示例代码
2019/08/04 Javascript
解决vue项目 build之后资源文件找不到的问题
2020/09/12 Javascript
vue 数据双向绑定的实现方法
2021/03/04 Vue.js
[36:54]Mineski vs Winstrike 2018国际邀请赛小组赛BO2 第一场 8.16
2018/08/17 DOTA
[02:23]1个至宝=115个英雄特效 最“绿”至宝拉比克“魔导师密钥”登场
2018/12/29 DOTA
[48:48]完美世界DOTA2联赛PWL S3 Magama vs GXR 第一场 12.19
2020/12/24 DOTA
用Python的线程来解决生产者消费问题的示例
2015/04/02 Python
Python中跳台阶、变态跳台阶与矩形覆盖问题的解决方法
2018/05/19 Python
修复 Django migration 时遇到的问题解决
2018/06/14 Python
Python面向对象程序设计之继承、多态原理与用法详解
2020/03/23 Python
应届大学生自荐信格式
2013/09/21 职场文书
教师自荐书
2013/10/08 职场文书
策划助理岗位职责
2013/11/18 职场文书
领导的自我鉴定
2013/12/28 职场文书
我的动漫时代的创业计划书范文
2014/01/27 职场文书
财务支持类个人的自我评价
2014/02/14 职场文书
促销活动方案模板
2014/02/24 职场文书
英文演讲稿
2014/05/15 职场文书
小学生常见病防治方案
2014/06/06 职场文书
2014幼儿园中班工作总结
2014/11/10 职场文书
Apache Kafka 分区重分配的实现原理解析
2022/07/15 Servers