python之 __enter__和__exit__

十四 __enter__和__exit__

我们知道在操作文件对象的时候可以这么写

with open('a.txt') as f:
     '代码块'

上述叫做上下文管理协议,即with语句,为了让一个对象兼容with语句,必须在这个对象的类中声明__enter__和__exit__方法

__exit__()中的三个参数分别代表异常类型,异常值和追溯信息,with语句中代码块出现异常,则with后的代码都无法执行

如果__exit()返回值为True,那么异常会被清空,就好像啥都没发生一样,with后的语句正常执行

# coding=utf-8
class Open:
    def __init__(self,name):
        self.name = name

    def __enter__(self):
        print('出现with语句,对象的__enter__被触发,有返回值则赋值给as声明的变量')
        return self

    def __exit__(self, exc_type, exc_val, exc_tb):
        print('with中代码块执行完毕时执行我啊')
        print(exc_type)
        print(exc_val)
        print(exc_tb)
        #return True

with Open('aa.txt') as f:
    print(f)
    print(f.name)
    print(sdsd)  # 触发 __exit__ ,之后的代码就不会运行
    print('-------')

print('End....')

'''
运行结果:
出现with语句,对象的__enter__被触发,有返回值则赋值给as声明的变量
<__main__.Open object at 0x0000016D7FA95BE0>
aa.txt
with中代码块执行完毕时执行我啊
<class 'NameError'>
name 'sdsd' is not defined
<traceback object at 0x0000016D7FA925C8>
End....
'''


'''
总结:
with obj as f:
    代码块
1. with obj ----> 触发obj.__enter__(),拿到返回值
2. as f --> f = 返回值
3. with obj as f 等同于  f = obj.__enter__()
4. 执行代码块
 1> 没有异常,整个代码块运行完毕后去触发__exit__(),它的三个参数都返回None
 2> 有异常的情况下,从异常出现的位置直接触发__exit))
    a:如果__exit__返回值为True,代表吞掉了异常
    b:如果__exit__返回值不为True,代表吐出了异常
    c:__exit__运行完毕,就代表整个with语句运行完毕
'''

用途或者说好处:

1.使用with语句的目的就是把代码块放入with中执行,with结束后,自动完成清理工作,无须手动干预

2.在需要管理一些资源比如文件,网络连接和锁的编程环境中,可以在__exit__中定制自动释放资源的机制,你无须再去关系这个问题,这将大有用处

  • 0
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
__enter__方法是上下文管理器协议中的一个方法,用于在进入with语句块之前执行一些操作。在引用\[1\]中的例子中,Test类实现了__enter__方法,在进入with语句块之前打开了一个文件,并返回了文件对象。这样,在with语句块中可以使用这个文件对象进行操作。\[1\] __enter__方法的返回值可以通过as关键字绑定到一个变量上,以便在with语句块中使用。在引用\[2\]的例子中,Open类的__enter__方法返回了一个字符串"enter返回值",在with语句块中可以通过变量f来引用这个返回值。\[2\] 使用上下文管理器的好处是可以确保资源的正确释放。在引用\[3\]的例子中,如果with语句块中出现异常,__exit__方法会被调用来处理异常并释放资源。这样可以避免资源泄漏。\[3\] 总之,使用上下文管理器可以简化资源的管理,确保资源的正确释放,并提高代码的可读性和可维护性。 #### 引用[.reference_title] - *1* [python的with和__enter__ 、 __exit__](https://blog.csdn.net/weixin_33851177/article/details/86028395)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [Python上下文管理协议 __enter__ 和 __exit__](https://blog.csdn.net/Python_222/article/details/129880336)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值