|
马上注册,结交更多好友,享用更多功能^_^
您需要 登录 才可以下载或查看,没有账号?立即注册
x
我就想问下c1.say_hi()和c2.say_hey()是怎么实现的?
当c1和c2作为Check类的实例对象时,访问say_hi()和say_hey()是作为属性访问还是方法访问,触发的getattr方法?
- class Check:
- def __init__(self, cls):
- self.cls = cls
- def __call__(self, *args, **kwargs):
- self.obj = self.cls(*args, **kwargs)
- return self
- def __getattr__(self, item):
- print(f'正在访问{item}....')
- return getattr(self.obj, item)
- @Check
- class C:
- def __init__(self, name):
- self.name = name
- def say_hi(self):
- print(f'hi~ -> {self.name}')
- def say_hey(self):
- print(f'hey~ -> {self.name}')
- c1 = C('c1')
- c2 = C('c2')
- c1.say_hi()
- # 正在访问say_hi....
- # hi~ -> c2
- c2.say_hey()
- # 正在访问say_hey....
- # hey~ -> c2
- # c1的name属性被c2的name属性覆盖
复制代码
在这个例子中,c1.say_hi()和c2.say_hey()都是方法访问。当你尝试访问这些方法时,会触发Check类的__getattr__方法。
这是因为当你使用装饰器 @Check 时,它实际上创建了一个 Check 类的实例,并将 C 类作为参数传递给它。
因此,当你调用 C('c1') 时,它实际上调用了 Check.__call__ 方法,并返回了一个 Check 类的实例。
由于 say_hi() 和 say_hey() 方法并不存在于 Check 类中,所以当你尝试访问它们时,会触发 Check.__getattr__ 方法。
在这个方法中,它打印出一条消息并返回了对应的方法。
值得注意的是,在这个例子中存在一个问题:每次调用被装饰的类(即 C)的构造函数时,都会创建一个新的 Check 实例,并且每次都会覆盖 self.obj 的值。
因此,在最后一次调用 C(‘c2’) 之后,self.obj 的值被设置为 c2 的实例对象。所以无论你调用 c1.say_hi() 还是 c2.say_hey() ,都会使用 c2 的 name 属性。
|
|