python学习之 getattr vs __get__ vs __getattr __ vs __getattribute__ vs __getitem__

 1. getattr、setattr、hasattr

  getattr比较常用,与setattr和hasattr一起出现,他们也是最容易理解的,下面是他的用法:

class Profile():
     name="xiaoxin"
def sex(self): return "male"p=Profile() hasattr(p, "name") # 判断属性是否存在 >>> True hasattr(p, "age") # 判断属性是否存在 >>> False
getattr(p, "name") # 获取属性值
>>> xiaoxin
getattr(p, "sex")
>>> <bound method Profile.sex of <__main__.Profile object at 0x7f22608f9710>>
getattr(p, "age", 14) # 如果属性不存在,则返回默认值
>>> 14
setattr(p, "age", "26") # 为属性赋值,并没有返回值 hasattr(p, "age") # 属性存在了 >>> True

2. __get__ 、 __set__、 __delete__

  提起__get__, 就不能不说  __set__, __delete__ , 一个类,只要其内部定义了方法 __get__, __set__, __delete__ 中的一个或多个,就可以称其为描述符。如果同时定义了__get__()__set__(),其被认为是一个数据描述符。只定义__get__()的描述符被称为非数据描述符。

  描述符常被用来作类型检查, 下面看它的具体用法:

class Integer:
    def __init__(self, name):
        self.name = name

    def __get__(self, instance, owner):
    
if instance is None: return self else: return instance.__dict__[self.name] def __set__(self, instance, value):
    
if not isinstance(value, int): raise TypeError('excepted an int') instance.__dict__[self.name] = value def __delete__(self, instance):
    
del instance.__dict__[self.name] class Point: x = Integer('x') y = Integer('y') def __init__(self, x, y): self.x = x self.y = y

p = Point(2, 3)
p.x
>> 2
p.y
>> 3
p = Point(2.5, 3)
>> TypeError: excepted an int

  需要注意的是,对象属性的访问顺序:实例属性>类属性>父类属性>__getattr__(),当Python解释器发现实例对象的字典中,有与描述符同名的属性时,描述符优先,会覆盖掉实例属性。

 3. __getattr__ 、__setattr__ 、__delattr__

  从对象中读取某个属性时,首先需要从self.__dicts__中搜索该属性,如果__dict__中没有该属性, 再从__getattr__中拿到返回值。当设置属性时,则会触发__setattr__方法,在这里可以进行类型验证(描述符只会对定义的描述符属性进行类型验证, 而__setattr__会对所有属性进行类型验证)

class Point:

    def __init__(self, x, y):
        self.x = x
        self.y = y

    def __getattr__(self, item):
        print(f'getattr {item}')
        return None

    def __setattr__(self, key, value):
        print(f'setattr {key}:{value}')
        if isinstance(value, int):
            return object.__setattr__(self, key, value)

    def __delattr__(self, item):
        print(f"delattr {item}")
        del self.__dict__[item]

p = Point(2, 3)

>>> setattr x:2
>>> setattr y:3

p.x

>>> 3

p.c

>>> getattr c

>>> None

p.c = 9

>>> setattr c:9

p.c

>>> 9

4. __getattribute__ 

   无论调用属性还是方法,都是先强制调用 __getattribute__ 方法,然后再传回属性的值或者是 函数(方法)的引用。

class Point:

    def __init__(self, x, y):
        self.x = x
        self.y = y

    def __getattribute__(self, item):   # 在 __getattribute__ 方法里面不要在出现self.**这种调用,因为每次调用类的属性都会强制调用 __getattribute__ ,会造成递归调用
        return 'i get you ' if item == 'x' else object.__getattribute__(self, item)

    def __getattr__(self, item):
        print(f'getattr {item}')
        return None
p = Point(2, 3)
p.x
>>> i get you
p.y
>>> 3
p.c # 首先调用 __getattribute__ , 里面实现了 查找__dict__里面有没有这个键,如果没有再调用 __getattr__方法
>>> getattr c
>>> None

 5. __getitem__、__setitem__、__delitem__

这三个方法主要用于对集合的操作

可变集合需要实现: __len__  __getitem__    __setitem__  __delitem__
不可变集合需要实现: __len__  __getitem__
__len__:返回集合长度
__getitem__(self, item) 使用索引访问元素
__setitem__(self, key, value) 对索引赋值,使用 self[key] = value 。
__delitem__(self, key) 删除索引值 del self[key]
__contains__ 实现in运算符,如果没有实现这个方法python也会调用__getitem__来使in运算符可用

class TemTest:

    def __init__(self,):
        self.x = [i for i in range(10)]

    def __len__(self):
        return len(self.x)

    def __getitem__(self, item):return self.x[item]

    def __setitem__(self, key, value):
        self.x[key] = value

    def __delitem__(self, key):
        del self.x[key]

    def __contains__(self, item):
        return item in self.x

    def __repr__(self):
        return '{}'.format(self.x)
test=TemTest() #实例化
print(len(test)) #返回长度
print(test[0])   #打印下标0的值
print(test[:3])  #切片
test[3]=10       #将下标3的值替换为10
print(test)
del test[3]      #删除下标3的值
print(test)
print(1 in test)  #测试in运算符
print(3 in test)  
10
0
[0, 1, 2]
[0, 1, 2, 10, 4, 5, 6, 7, 8, 9]
[0, 1, 2, 4, 5, 6, 7, 8, 9]
True
False

参考博文:https://www.cnblogs.com/flashBoxer/p/9645939.html 

转载于:https://www.cnblogs.com/jiaxiaoxin/p/10741262.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
`__getattr__`和`__getattribute__`是Python中特殊的方法,用于处理对象的属性访问和获取。 `__getattr__`用于在对象实例的属性不存在时被调用,通常在动态获取属性的情况下使用。当我们尝试访问一个对象上不存在的属性时,Python解释器会自动调用`__getattr__`方法。我们可以在这个方法中定义对不存在属性的处理逻辑,比如返回默认值或者触发异常。 下面是一个示例: ``` class MyClass: def __getattr__(self, name): return f"Attribute {name} does not exist." obj = MyClass() print(obj.some_attr) # Output: Attribute some_attr does not exist. ``` `__getattribute__`用于在对象实例的属性被访问时被调用。与`__getattr__`不同,`__getattribute__`会在所有属性访问上被调用,无论属性是否存在。我们可以通过在`__getattribute__`方法中自定义属性访问的行为,比如在访问属性前执行一些操作或返回特定的值。 下面是一个示例: ``` class MyClass: def __getattribute__(self, name): print(f"Accessing attribute {name}") return super().__getattribute__(name) obj = MyClass() print(obj.some_attr) # Output: # Accessing attribute some_attr # None ``` 注意,使用`__getattribute__`时需要谨慎,因为在方法内部访问同个实例的其他属性时可能会触发无限递归调用。在这种情况下,可以通过使用`super()`来避免无限递归。 综上所述,`__getattr__`和`__getattribute__`是Python中用于处理对象属性访问和获取的特殊方法,分别用于处理不存在属性和所有属性的访问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值