python自定义属性访问__setattr__, __getattr__, __setattribute__

object._getattr_(self, name)

实例instance通过instance.name访问属性name,只有当属性name没有在实例的__dict__或它构造类的__dict__或基类的__dict__中没有找到,才会调用__getattr__。当属性name可以通过正常机制追溯到时,__getattr__是不会被调用的。如果在__getattr__(self, attr)存在通过self.attr访问属性,会出现无限递归错误。

class ClassA(object):

    def __init__(self, classname):
        self.classname = classname

    def __getattr__(self, attr):
        return(‘invoke __getattr__‘, attr)

insA = ClassA(‘ClassA‘)
print(insA.__dict__) # 实例insA已经有classname属性了
# {‘classname‘: ‘ClassA‘}

print(insA.classname) # 不会调用__getattr__
# ClassA

print(insA.grade) # grade属性没有找到,调用__getattr__
# (‘invoke __getattr__‘, ‘grade‘)

object.__getattribute__(self, name)

实例instance通过instance.name访问属性name__getattribute__方法一直会被调用,无论属性name是否追溯到。如果类还定义了__getattr__方法,除非通过__getattribute__显式的调用它,或者__getattribute__方法出现AttributeError错误,否则__getattr__方法不会被调用了。如果在__getattribute__(self, attr)方法下存在通过self.attr访问属性,会出现无限递归错误。

如下所示,ClassA中定义了__getattribute__方法,实例insA获取属性时,都会调用__getattribute__返回结果,即使是访问__dict__属性。

class ClassA(object):

    def __init__(self, classname):
        self.classname = classname

    def __getattr__(self, attr):
        return(‘invoke __getattr__‘, attr)

    def __getattribute__(self, attr):
        return(‘invoke __getattribute__‘, attr)

insA = ClassA(‘ClassA‘)
print(insA.__dict__)
# (‘invoke __getattribute__‘, ‘__dict__‘)

print(insA.classname)
# (‘invoke __getattribute__‘, ‘classname‘)

print(insA.grade)
# (‘invoke __getattribute__‘, ‘grade‘)

object.__setattr__(self, name, value)

如果类自定义了__setattr__方法,当通过实例获取属性尝试赋值时,就会调用__setattr__

常规的对实例属性赋值,被赋值的属性和值会存入实例属性字典__dict__中。

class ClassA(object):

    def __init__(self, classname):
        self.classname = classname

insA = ClassA(‘ClassA‘)

print(insA.__dict__)
# {‘classname‘: ‘ClassA‘}

insA.tag = ‘insA‘    

print(insA.__dict__)
# {‘tag‘: ‘insA‘, ‘classname‘: ‘ClassA‘}

如下类自定义了__setattr__,对实例属性的赋值就会调用它。类定义中的self.attr也同样,所以在__setattr__下还有self.attr的赋值操作就会出现无线递归的调用__setattr__的情况。自己实现__setattr__有很大风险,一般情况都还是继承object类的__setattr__方法。

class ClassA(object):
    def __init__(self, classname):
        self.classname = classname

    def __setattr__(self, name, value):
        # self.name = value  # 如果还这样调用会出现无限递归的情况
        print(‘invoke __setattr__‘)

insA = ClassA(‘ClassA‘) # __init__中的self.classname调用__setattr__。
# invoke __setattr__

print(insA.__dict__)
# {}

insA.tag = ‘insA‘
# invoke __setattr__

print(insA.__dict__)
# {}

object.__delattr__(self, name)

Like __setattr__() but for attribute deletion instead of assignment. This should only be implemented if del obj.name is meaningful for the object.

object.__dir__(self)

dir()作用在一个实例对象上时,__dir__会被调用。返回值必须是序列。dir()将返回的序列转换成列表并排序。


object.__call__(self[, args...])

Called when the instance is “called” as a function; if this method is defined, x(arg1, arg2, ...) is a shorthand for x.__call__(arg1, arg2, ...).

Python中有一个有趣的语法,只要定义类型的时候,实现__call__函数,这个类型就成为可调用的。换句话说,我们可以把这个类的对象当作函数来使用,相当于重载了括号运算符。

class Student(object):
    def __init__(self, name):
        self.name = name
    def __call__(self):
        print(‘My name is %s.‘ % self.name)

s = Student(‘Michael‘)
s()
# My name is Michael.


通过使用__setattr__, __getattr__, __delattr__可以重写dict,使之通过“.”调用键值。

class Dict(dict):
    ‘‘‘
    通过使用__setattr__,__getattr__,__delattr__
    可以重写dict,使之通过“.”调用
    ‘‘‘
    def __setattr__(self, key, value):
        print("In ‘__setattr__")
        self[key] = value

    def __getattr__(self, key):
        try:
            print("In ‘__getattr__")
            return self[key]
        except KeyError as k:
            return None

    def __delattr__(self, key):
        try:
            del self[key]
        except KeyError as k:
            return None

    # __call__方法用于实例自身的调用,达到()调用的效果
    def __call__(self, key):    # 带参数key的__call__方法
        try:
            print("In ‘__call__‘")
            return self[key]
        except KeyError as k:
            return "In ‘__call__‘ error"

s = Dict()
print(s.__dict__)
# {}

s.name = "hello"    # 调用__setattr__
# In ‘__setattr__

print(s.__dict__) # 由于调用的‘__setattr__‘, name属性没有加入实例属性字典中。
# {}

print(s("name"))    # 调用__call__
# In ‘__call__‘
# hello

print(s["name"])    # dict默认行为
# hello

# print(s)
print(s.name)       # 调用__getattr__
# In ‘__getattr__
# hello

del s.name          # 调用__delattr__
print(s("name"))    # 调用__call__
# None

原文地址:https://www.cnblogs.com/zhangyue123/p/9738097.html

时间: 2024-08-10 15:11:32

python自定义属性访问__setattr__, __getattr__, __setattribute__的相关文章

python __setattr__, __getattr__, __delattr__, __call__

python __setattr__, __getattr__, __delattr__, __call__ getattr `getattr`函数属于内建函数,可以通过函数名称获取 value = obj.attribute value = getattr(obj, "attribute") 使用`getattr`来实现工厂模式 #一个模块支持html.text.xml等格式的打印,根据传入的formate参数的不同,调用不同的函数实现几种格式的输出 import statsout

17、类的自定义属性访问及动态属性设置

前言:本文主要介绍python类的一些自定义属性访问的方法,以及类的动态属性设置即python的内置函数setattr(). 自定义属性访问 什么是属性?下面的例子a和b是属性吗?不是,他们是全局变量,属性(attribute)是类中的成员变量,也可以理解为属性就是类变量. a = 11234 b = 'python' 类中的变量是静态变量,类可以直接访问,python是一门动态语言,任何实例对象都可以动态地添加或删除属性,一个类定义了一个作用域,类实例对象也引入了一个作用域,这与类定义的作用域

pyinstaller打包python源程序访问hive

1.需求 使用hvie server一段时间后,业务部门需要自己不定时的查询业务数据,之前这一块都是他们提需求我们来做,后来发现这样重复一样的工作放在我们这边做是在没有效率,遂提出给他们工具或者web UI自助查询,当然hive有自己的hwi可以通过网页UI进行自助查询,但是这对不懂sql的业务人员有点不太友好,目前有没时间去修改hwi的UI,所以还是给他们提供查询工具吧.我这边主要使用python thrift访问集群的hive,所以自然要将python源码打包成.exe,业务人员在windo

python __setattr__, __getattr__, __delattr__,__getattribute__

参考资料 __setattr__.__getattr__和__delattr__以及__getattribute__可以拦截对对象属性的访问: >>> s = Something() >>> s.age = 3 set 'age' = 3 >>> s.age 3 注意到,s.age并没有调用__getattr__,是因为__getattr__方法只在属性没有找到的时候调用. 虽然相比于Property,实现优点复杂(可能效率更低?),但是足够灵活,功能

python属性访问

#! /usr/bin/env python #! -*- cording:utf-8 -*- # # class C: # def __getattribute__(self, name): # print("getattribute") # return super().__getattribute__(name)#super默认找父类 # def __getattr__(self, name): # print("getattr") # def __setat

python中__get__ vs __getattr__ vs __getattribute__以及属性的搜索策略

其中,__getattribute__是无条件被调用. 对任何对象的属性访问时,都会隐式的调用__getattribute__方法,比如调用t.__dict__,其实执行了t.__getattribute__("__dict__")函数.所以如果我们在重载__getattribute__中又调用__dict__的话,会无限递归,用object大神来避免,即object.__getattribute__(self, name). 只有__getattribute__找不到的时候,才会调用

自定义属性访问

__getattr__属性 什么时候触发: 当我们访问属性的时候,如果属性不存在,则触发此方法: class MyTest: def __getattr__(self, item): # 当访问的属性不存在时,该方法会被触发 print("触发__getattr__属性") super().__getattribute__(item) mt = MyTest() mt.name 执行结果: mt.name File "C:/Users/Administrator/Pychar

python SSL访问网站代码

经常遇到要用Python访问SSL网站,这是一个简单的代码,几行就搞定了! import socket import ssl sock = ssl.wrap_socket(socket.socket()) sock.connect(('www.test.com', 443))#其中www.test.com要访问的网站 data = 'xxxxx'#data为要发送的数据 sock.sendall(data)recv_data = sock.recv(8192) sock.close() prin

Python脚本访问Greenplum数据库安装指导

安装前准备 (1)操作系统(系统上面要安装一些必备的开发工具(比如gcc等)) linux-82:/home/PyODBC # cat/etc/SuSE-release SUSE Linux EnterpriseServer 11 (x86_64) VERSION = 11 PATCHLEVEL = 1 (2)安装所需的软件包 greenplum-connectivity-4.3.0.0-build-2-SuSE10-x86_64.zip --GP官网下载,GP的JDBC和ODBC驱动 pyod