zoukankan      html  css  js  c++  java
  • 继承的方式完成包装__attr__

    __getattr__    当实例要调用的属性不存在的时候会触发

    __setattr__    当实例要设置属性的时候会触发

    __delattr__    当实例要删除属性的时候会触发

    这三个方法是python内置的方法,如果不自己定义的话,就是默认的,自己定义的话就是用自己定义的

    1、__getattr__           当要调用的属性不存在的时候会触发

    没有自己定义__getattr__的时候

    class Foo:
        def __init__(self,name):
            self.name=name
    f1=Foo("gouguoqi")
    print(f1.name)
    
    C:python35python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
    
    gouguoqi

    我们自己定义一个__getattr__

    class Foo:
        def __init__(self,name):
            self.name=name
        def __getattr__(self, item):
            print("你要找的属性%s不存在" %item)
    f1=Foo("gouguoqi")
    print(f1.name)
    print(f1.age)
    
    C:python35python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
    
    gouguoqi
    
    你要找的属性age不存在

    2、__setaddr__    给属性设置值的时候会触发

    class Foo:
        def __init__(self,name):
            self.name=name
        def __setattr__(self, k, v):
            print("执行setattr")
    f1=Foo("gouguoqi")#实例化的过程就是运行init,init就有一个赋值name=gouguoqi所以就触发setaddr
    
    C:python35python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
    
    执行setattr

    2.1我们在没有自己定义__setattr__的时候,系统是帮我们把属性设置到属性字典里面

    class Foo:
        def __init__(self,name):
            self.name=name
    f1=Foo("gouguoqi")#实例化的过程就是运行init,init就有一个赋值name=gouguoqi所以就触发setaddr
    print(f1.__dict__)
    
    C:python35python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
    
    {'name': 'gouguoqi'}

    2.2但是我们自己定制的那个setattr只是执行了一个打印的操作,并没有帮我们把值新增到属性字典里面去

    class Foo:
        def __init__(self,name):
            self.name=name
        def __setattr__(self, k, v):
            print("执行setattr")
    f1=Foo("gouguoqi")#实例化的过程就是运行init,init就有一个赋值name=gouguoqi所以就触发setaddr
    f1.age=28
    f1.gender="male"
    print(f1.__dict__)
    
    C:python35python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
    
    执行setattr
    
    执行setattr
    
    执行setattr
    
    {} 

    2.3现在有一个需求,传入的值必须是字符串,应该怎么实现

    先完成以下判断逻辑,然后在具体的去实现功能

    class Foo:
        def __init__(self,name):
            self.name=name
        def __setattr__(self, k, v):
            print("执行setattr")
            if type(v) is str:
                print("开始设置")
            else:
                print("必须是字符串")
    f1=Foo("gouguoqi")
    f1.age=28
    print(f1.__dict__)
    
    C:python35python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
    
    执行setattr
    
    开始设置
    
    执行setattr
    
    必须是字符串
    
    {}

    实现设置功能,只能是字符串才能设置进去

    class Foo:
        def __init__(self,name):
            self.name=name
        def __setattr__(self, k, v):
            print("执行setattr")
            if type(v) is str:
                print("开始设置")
                self.__dict__[k]=v
            else:
                print("必须是字符串")
    f1=Foo("gouguoqi")
    f1.age=28
    print(f1.__dict__)
    
     
    
    C:python35python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
    
    执行setattr
    
    开始设置
    
    执行setattr
    
    必须是字符串
    
    {'name': 'gouguoqi'}

    2.4还可以实现。如果是字符串的话,把字符串变为大写

    class Foo:
        def __init__(self,name):
            self.name=name
        def __setattr__(self, k, v):
            print("执行setattr")
            if type(v) is str:
                print("开始设置")
                self.__dict__[k]=v.upper()
            else:
                print("必须是字符串")
    f1=Foo("gouguoqi")
    f1.age=28
    print(f1.__dict__)
    
    C:python35python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
    
    执行setattr
    
    开始设置
    
    执行setattr
    
    必须是字符串
    
    {'name': 'GOUGUOQI'}

    3、__delattr__  删除属性

    class Foo:
        def __init__(self,name):
            self.name=name
        def __delattr__(self, item):
            print("执行delattr",item)
            self.__dict__.pop(item)#操作底层的属性字典的key
    f1=Foo("gouguoqi")
    del f1.name#把f1传给self,把name传给item
    print(f1.__dict__)
    
    C:python35python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
    
    执行delattr name
    
    {}

    3.1那么系统默认的删除就是这样的啊,我们为什么要自己定义一个这个呢,那么又来一个新的需求,所有的属性都不能删除,改怎么做呢

    class Foo:
        def __init__(self,name):
            self.name=name
        def __delattr__(self, item):
            print("%s属性不允许删除" %item)
    f1=Foo("gouguoqi")
    del f1.name
    print(f1.__dict__)
    
    C:python35python3.exe D:/pyproject/day26/双下划线开头的attr方法.py
    
    name属性不允许删除
    
    {'name': 'gouguoqi'

    4、包装 

    class List(list):#定义个类,继承系统的list
        pass
    
    l2=list("hello world")
    print(l2,type(l2))
    
    l1=List("hello world")
    print(l1,type(l1))
    
    C:python35python3.exe D:/pyproject/day26/包装标准类型.py
    
    ['h', 'e', 'l', 'l', 'o', ' ', 'w', 'o', 'r', 'l', 'd'] <class 'list'>
    
    ['h', 'e', 'l', 'l', 'o', ' ', 'w', 'o', 'r', 'l', 'd'] <class '__main__.List'>

    4.1现在想要实现的是 定制一个方法,显示出列表中间元素的那个元素内容

    class List(list):#定义个类,继承系统的list
        def show_middle(self):
            mid_index=int(len(self)/2)#取到中间的索引
            return self[mid_index]#返回列表的中间的索引的那个值
    l1=List("helloworld")
    print(l1,type(l1))
    print(l1.show_middle())
    
    C:python35python3.exe D:/pyproject/day26/包装标准类型.py
    
    ['h', 'e', 'l', 'l', 'o', 'w', 'o', 'r', 'l', 'd'] <class '__main__.List'>
    
    w

    4.2我们自己定义一个apped的方法

    结果就是append这个方法不能追加了。因为自己定义的是pass

    class List(list):#定义个类,继承系统的list
        def append(self, object):
            pass
        def show_middle(self):
            mid_index=int(len(self)/2)#取到中间的索引
            return self[mid_index]#返回列表的中间的索引的那个值
    l1=List("helloworld")
    l1.append(111)
    print(l1)
    
    C:python35python3.exe D:/pyproject/day26/包装标准类型.py
    
    ['h', 'e', 'l', 'l', 'o', 'w', 'o', 'r', 'l', 'd']

    实现真正的append操作

    class List(list):#定义个类,继承系统的list
        def append(self, object):
            if type(object) is str:
                list.append(self,object)#用父类的append方法
        def show_middle(self):
            mid_index=int(len(self)/2)#取到中间的索引
            return self[mid_index]#返回列表的中间的索引的那个值
    l1=List("helloworld")
    l1.append("111")
    print(l1)
    
    C:python35python3.exe D:/pyproject/day26/包装标准类型.py
    
    ['h', 'e', 'l', 'l', 'o', 'w', 'o', 'r', 'l', 'd', '111']

    类调用方法,推荐用super

    class List(list):#定义个类,继承系统的list
        def append(self, object):
            if type(object) is str:
                # list.append(self,object)#用父类的append方法
                super().append(object)#super就是调用父类的方法,不用传self
        def show_middle(self):
            mid_index=int(len(self)/2)#取到中间的索引
            return self[mid_index]#返回列表的中间的索引的那个值
    l1=List("helloworld")
    l1.append("222")
    print(l1)
    
    C:python35python3.exe D:/pyproject/day26/包装标准类型.py
    
    ['h', 'e', 'l', 'l', 'o', 'w', 'o', 'r', 'l', 'd', '222']

    4.3这样就实现了  append我们自己定制了一个,只能添加字符串,不能append数字

    class List(list):#定义个类,继承系统的list
        def append(self, object):
            if type(object) is str:
                # list.append(self,object)#用父类的append方法
                super().append(object)#super就是调用父类的方法,不用传self
            else:
                print("只能append字符串")
        def show_middle(self):
            mid_index=int(len(self)/2)#取到中间的索引
            return self[mid_index]#返回列表的中间的索引的那个值
    l1=List("helloworld")
    l1.append("222")
    print(l1)
    l1.append(666)
    
    C:python35python3.exe D:/pyproject/day26/包装标准类型.py
    
    ['h', 'e', 'l', 'l', 'o', 'w', 'o', 'r', 'l', 'd', '222']

    只能append字符串

  • 相关阅读:
    简道云--编辑应用入门
    虚拟机迁移
    KVM虚拟化网络管理(一)
    keepalived概述
    kvm虚拟化存储管理
    免密登录gitlab
    KVM虚拟化介绍
    Jenkins+Git+Gitlab+Ansible实现持续集成自动化部署静态网站
    Jenkins+Git+Gitlab+Ansible实现持续集成自动化部署动态网站
    Jenkins凭证介绍
  • 原文地址:https://www.cnblogs.com/gouguoqilinux/p/9215536.html
Copyright © 2011-2022 走看看