zoukankan      html  css  js  c++  java
  • 面向对象进阶5:上下文管理协议

    一、 __enter__和__exit__

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

    with为工厂函数,open得到的就是一个类,赋值为f。

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

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

    #上下文管理协议
    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中代码块执行完毕时执行我啊')
    
    with Open('a.txt') as f:      # 并不是把Open('a.txt')结果返回给f,而是触发Open中的enter方法,将enter函数的返回值给f
        print('=====>执行代码块')  
       print(f)  #<__main__.Foo object at 0x0000000001199198>  #说明f为一个对象    
    print(f.name) #可以利用f来调用它当中的变量name #等待with对应代码块执行结束,触发exit函数

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

    class Foo:
        def __init__(self,name):
            self.name=name
    
        def __enter__(self):
            print('执行enter')
            return self
    
        def __exit__(self, exc_type, exc_val, exc_tb):
            print('执行exit')
            print(exc_type)   #None
            print(exc_val)      #None
            print(exc_tb)      #None
            return True
    with Foo('a.txt') as f:
        print(f)
        print(asddfasdfasdfasfdasfd)  #这是一个不存在的变量名
        #出现异常,直接触发__exit__  #报错:name error,
        #执行exit之后整个程序的所有内容都不执行了
        #如果exit中返回一个True,异常信息不会被打印出来,with中内容不会执行
        #直接跳出with代码块,with代码块外的程序现在就可以执行了
        print(f.name)
        print('-----------------')
        print('-----------------')
        print('-----------------')
        print('-----------------')
        print('-----------------')
        print('-----------------')
        print('-----------------')
    print('000000000000000000000000000000000000000000000')

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

    #模拟open
    class Open:
        def __init__(self,filepath,mode='r',encoding='utf-8'):
            self.filepath=filepath
            self.mode=mode
            self.encoding=encoding
    
        def __enter__(self):
            # print('enter')
            self.f=open(self.filepath,mode=self.mode,encoding=self.encoding)
            return self.f
    
        def __exit__(self, exc_type, exc_val, exc_tb):
            # print('exit')
            self.f.close()
            return True 
        def __getattr__(self, item):
            return getattr(self.f,item)
    
    with Open('a.txt','w') as f:
        print(f)
        f.write('aaaaaa')
        f.wasdf #抛出异常,交给__exit__处理
    # 上下文管理协议总结
    with obj as  f:
        '代码块'
        
    1.with obj ----》触发obj.__enter__(),拿到返回值
    
    2.as f----->f=enter函数返回值、
    
    3.with obj as f  等同于     f=obj.__enter__()
    
    4.执行代码块
    一:没有异常的情况下,整个代码块运行完毕后去触发__exit__,它的三个参数都为None
    二:有异常的情况下,从异常出现的位置直接触发__exit__
        a:如果__exit__的返回值为True,代表吞掉了异常,with语句块结束,继续执行with外面的剩下程序语句
        b:如果__exit__的返回值不为True,代表吐出了异常,整个程序都不执行; 
        c:__exit__的的运行完毕就代表了整个with语句的执行完毕
        
    因此要实现with运行结束后自动清理的功能,应该将清理功能放在在exit函数中

    用途或者说好处:

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

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

    #补充异常
    例子:在cmd中输入一个未曾定义过的变量名
    追踪信息Traceback:(most recent call last):
            File "<stdin>":line 1. in<module>
    对应的就是exit函数参数表中的ext_tb
    
    异常类ext_type:异常值ext_val
    NameError:name 'addfsj' is not defined
  • 相关阅读:
    android开发布局三(微信布局)
    Android开发adb,SQLite数据库运用
    直线电机磁负荷、电负荷
    MIT公开课(一):电场和偶极子
    哈工大电气工程系硕士研究生入学复试——自动控制原理1、2章
    直线电机与旋转电机的区别
    Math类介绍
    Scala编辑器和IntelliJ IDEA开发环境配置
    减少cpu的方法
    AS内存清理,建议以及查找内存泄露的方法
  • 原文地址:https://www.cnblogs.com/Josie-chen/p/8903492.html
Copyright © 2011-2022 走看看