zoukankan      html  css  js  c++  java
  • python cookbook第三版学习笔记十二:类和对象(三)创建新的类或实例属性

    先介绍几个类中的应用__getattr__,__setattr__,__get__,__set__,__getattribute__,。
    __getattr__:当在类中找不到attribute的时候,会调用__getattr__,并执行其中的自定义代码。所有在类中定义的属性都包含在__dict__中,也就是说如果在__dict__中找不到对应的属性名,则__getattr__被触发。
    class get_try(object):
        def __init__(self,value):
            self.value=value
        def __getattr__(self, item): #当找不到类中的属性的时候,将会被调用
            self.value=item

    if __name__ == "__main__":
        g=get_try('value')
        g.value1   #调用了g.value1,value1传入__getattr__的item.调用__getattr后,value=value1
        print g.value
    print g.__dict_
     
    E:python2.7.11python.exe E:/py_prj/fluent_python/chapter7.py
    value1
    {'value': 'value1'}
    结果打印的时候g.value的值是value1
     
    __getattribute__:无条件被调用,如果同时定义了__getattr__, 则__getattr__不会被调用。
    class get_try(object):
        def __init__(self,value):
            self.value=value
        def __getattr__(self, item):
            self.value=item
        def __getattribute__(self, item):
            print item

    if __name__ == "__main__":
        g=get_try('value')
        g.value1
    E:python2.7.11python.exe E:/py_prj/fluent_python/chapter7.py
    value1
    结果和之前一样。
     
    __setattr__当需要自己对属性进行定义的时候,会被调用。比如self.att=value就会变成self.__setattr__(“att”,value)class get_try():    def __init__(self,value):
            self.value=value
        def __getattr__(self, item):
            self.value=item
        def __getattribute__(self, item):
            print item
        def __setattr__(self, key, value):
            self.__dict__[key]=value


    if __name__ == "__main__":
        g=get_try('value')
        g.value1=3
        print g.value1
        print g.__dict__
    E:python2.7.11python.exe E:/py_prj/python_cookbook/chapter8.py
    3
    {'value1': 3, 'value': 'value'}
    需要注意的是在__setattr__的时候不能进行self.key=value的赋值,因为这个方式会使得不停调用self.__setattr__。这样会形成无穷的递归循环。最终导致堆栈溢出异常。当我们调用的时候,报的错误就像下面这种
      File "E:/py_prj/python_cookbook/chapter8.py", line 107, in __setattr__
        self.key=value
      File "E:/py_prj/python_cookbook/chapter8.py", line 107, in __setattr__
        self.key=value
    RuntimeError: maximum recursion depth exceeded
    其实我们不实现__setattr__的时候,在给属性赋值的时候也会隐含的调用这个函数。那么这样实现的好处是什么呢?好处是我们可以指定给哪些属性进行赋值。
    class get_try():
        def __init__(self,value):
            self.value=value
        def __getattr__(self, item):
            self.value=item
        def __getattribute__(self, item):
            print item
        def __setattr__(self, key, value):
            if key == 'value1':
                print 'incorrect key'
            else
    :
                self.__dict__[key]=value


    if __name__ == "__main__":
        g=get_try('value')
        g.value1=3
        g.value2=4
        print g.__dict__
    E:python2.7.11python.exe E:/py_prj/python_cookbook/chapter8.py
    incorrect key
    {'value2': 4, 'value': 'value'}
    上面的代码中,当对value1进行赋值的时候,首先进行判断。如果是value1则打印incorrect key。所以看到当对value2进行赋值的时候是成功的。同样的道理,我们可以用__setattr__在防止后续对类中已有的属性进行赋值:
    class get_try():
        def __init__(self,value):
            self.value=value
        def __getattr__(self, item):
            self.value=item
        def __getattribute__(self, item):
            print item
        def __setattr__(self, key, value):
            if key in self.__dict__:
                print 'already exist'
            else
    :
                self.__dict__[key]=value

    if __name__ == "__main__":
        g=get_try('value')
        g.value=4
        g.value1=3
        print g.__dict__
    E:python2.7.11python.exe E:/py_prj/python_cookbook/chapter8.py
    already exist
    {'value1': 3, 'value': 'value'}
    上面的代码中,首先判断赋值的对象是否已存在,如果存在则打印already exist
     
  • 相关阅读:
    MVC4中EasyUI Tree异步加载JSON数据生成树
    Linq语法
    委托的多种写法
    centos7安装Hive2.3.0
    ApacheKylin笔记
    把普通的JavaProject变成MavenProject
    eclipse 手动/自动安装插件
    我的JdbcUtils类
    Druid 连接池 JDBCUtils 工具类的使用
    Maven下载cxf所需要的jar包
  • 原文地址:https://www.cnblogs.com/zhanghongfeng/p/7220554.html
Copyright © 2011-2022 走看看