黎子卿 发表于 2023-8-29 16:46:21

为什么

class C:
    def __init__(self,name,age):
      self.name = name
      self.__age = age
    def __getattribute__(self,attrname):
      print("被拦截了")
      return super().__getattribute__(attrname)
定义了一个类C
然后
c = C("哈哈哈",18)
getattr(c,"name")
为什么会输出
被拦截了
“哈哈哈”
搞不懂这个重写的__getattribute__魔法方法对python自带的getattr函数起了什么作用。
为什么会打印“被拦截了”

isdkz 发表于 2023-8-29 16:46:46

在这段代码中,类C重写了 __getattribute__ 魔法方法。 __getattribute__ 方法在访问对象的属性时被调用。

当我们调用 getattr(c, "name") 时, getattr 函数会尝试获取对象 c 的属性 name 。由于类C重写了 __getattribute__ 方法,所以在获取属性之前,会先执行 print("被拦截了") 语句。

super().__getattribute__(attrname) 调用了父类的 __getattribute__ 方法,实际上是获取了对象的属性值。因此,最终输出的结果是 "哈哈哈" 。

重写 __getattribute__ 方法的作用是在获取对象属性时进行拦截和处理。在这个例子中,打印了一条信息表示属性被拦截了,然后继续获取属性的值并返回。这样可以在属性访问时添加额外的逻辑或者对属性进行保护。

isdkz 发表于 2023-8-29 17:54:40

看帮助文档可知 getattr(x, y) 相当于 x.y

而 x.y 是会去调用 __getattribute__ 的,因为你重写了 __getattribute__ ,所以会打印“被拦截了”

>>> help(getattr)
Help on built-in function getattr in module builtins:

getattr(...)
    getattr(object, name[, default]) -> value

    Get a named attribute from an object; getattr(x, 'y') is equivalent to x.y.
    When a default argument is given, it is returned when the attribute doesn't
    exist; without it, an exception is raised in that case.

>>>

cjjJasonchen 发表于 2023-8-29 17:57:08

你在getattr里面加了一个print,就这么简单{:10_262:}
页: [1]
查看完整版本: 为什么