设为首页 收藏本站
查看: 1081|回复: 0

[经验分享] Python 的 with 语句详解

[复制链接]

尚未签到

发表于 2015-12-15 09:34:29 | 显示全部楼层 |阅读模式
这篇文章主要介绍了Python 的 with 语句,本文详细讲解了with语句、with语句的历史、with语句的使用例子等,需要的朋友可以参考下  一、简介
  with是从Python 2.5 引入的一个新的语法,更准确的说,是一种上下文的管理协议,用于简化try…except…finally的处理流程。with通过__enter__方法初始化,然后在__exit__中做善后以及处理异常。对于一些需要预先设置,事后要清理的一些任务,with提供了一种非常方便的表达。
  with的基本语法如下,EXPR是一个任意表达式,VAR是一个单一的变量(可以是tuple),”as VAR”是可选的。



  • with EXPR as VAR:

  •     BLOCK
根据PEP 343的解释,with…as…会被翻译成以下语句:




  • mgr = (EXPR)

  • exit = type(mgr).__exit__  # Not calling it yet
  • value = type(mgr).__enter__(mgr)
  • exc = True
  • try:
  •     try:
  •         VAR = value  # Only if "as VAR" is present
  •         BLOCK
  •     except:
  •         # The exceptional case is handled here
  •         exc = False
  •         if not exit(mgr, *sys.exc_info()):
  •             raise
  •         # The exception is swallowed if exit() returns true
  • finally:
  •     # The normal and non-local-goto cases are handled here
  •     if exc:
  •         exit(mgr, None, None, None)
为什么这么复杂呢?注意finally中的代码,需要BLOCK被执行后才会执行finally的清理工作,因为当EXPR执行时抛出异常,访问mgr.exit执行就会报AttributeError的错误。  

  
二、实现方式
  根据前面对with的翻译可以看到,被with求值的对象必须有一个__enter__方法和一个__exit__方法。稍微看一个文件读取的例子吧,注意在这里我们要解决2个问题:文件读取异常,读取完毕后关闭文件句柄。用try…except一般会这样写:



  • f = open('/tmp/tmp.txt')

  • try:
  •     for line in f.readlines():
  •         print(line)
  • finally:
  •     f.close()
注意我们这里没有处理文件打开失败的IOError,上面的写法可以正常工作,但是对于每个打开的文件,我们都要手动关闭文件句柄。如果要使用with来实现上述功能,需要需要一个代理类:




  • class opened(object):

  •     def __init__(self, name):
  •         self.handle = open(name)
  •     def __enter__(self):
  •         return self.handle
  •     def __exit__(self, type, value, trackback):
  •         self.handle.close()
  • with opened('/tmp/a.txt') as f:
  •     for line in f.readlines():
  •         print(line)

注意我们定了一个名字叫opened的辅助类,并实现了__enter__和__exit__方法,__enter__方法没有参数,__exit__方法的3个参数,分别代表异常的类型、值、以及堆栈信息,如果没有异常,3个入参的值都为None。  

  如果你不喜欢定义class,还可以用Python标准库提供的contextlib来实现:




  • from contextlib import contextmanager

  • @contextmanager
  • def opened(name):
  •     f = open(name)
  •     try:
  •         yield f
  •     finally:
  •         f.close()
  • with opened('/tmp/a.txt') as f:
  •     for line in f.readlines():
  •         print(line)

使用contextmanager的函数,yield只能返回一个参数,而yield后面是处理清理工作的代码。在我们读取文件的例子中,就是关闭文件句柄。这里原理上和我们之前实现的类opened是相同的,有兴趣的可以参考一下contextmanager的源代码。  

  三、应用场景

废话了这么多,那么到底那些场景下该使用with,有没有一些优秀的例子?当然啦,不然这篇文章意义何在。以下摘自PEP 343。
  一个确保代码执行前加锁,执行后释放锁的模板:




  • @contextmanager

  •     def locked(lock):
  •         lock.acquire()
  •         try:
  •             yield
  •         finally:
  •             lock.release()
  •     with locked(myLock):
  •         # Code here executes with myLock held. The lock is
  •         # guaranteed to be released when the block is left (even
  •         # if via return or by an uncaught exception).
  

数据库事务的提交和回滚:




  • @contextmanager

  •         def transaction(db):
  •             db.begin()
  •             try:
  •                 yield None
  •             except:
  •                 db.rollback()
  •                 raise
  •             else:
  •                 db.commit()
重定向stdout:




  • @contextmanager

  • def stdout_redirected(new_stdout):
  •     save_stdout = sys.stdout
  •     sys.stdout = new_stdout
  •     try:
  •         yield None
  •     finally:
  •         sys.stdout = save_stdout
  • with opened(filename, "w") as f:
  •     with stdout_redirected(f):
  •         print "Hello world"

注意上面的例子不是线程安全的,再多线程环境中要小心使用。  

  
四、总结
  with是对try…expect…finally语法的一种简化,并且提供了对于异常非常好的处理方式。在Python有2种方式来实现with语法:class-based和decorator-based,2种方式在原理上是等价的,可以根据具体场景自己选择。
  with最初起源于一种block…as…的语法,但是这种语法被很多人所唾弃,最后诞生了with,关于这段历史依然可以去参考PEP-343和PEP-340

http://www.iyunv.net/article/51045.htm

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-151379-1-1.html 上篇帖子: Python异步非阻塞IO多路复用Select/Poll/Epoll使用 下篇帖子: python callable(object)
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表