gpt4 book ai didi

python - 使用类方法而不是同名的实例方法

转载 作者:太空宇宙 更新时间:2023-11-04 08:57:50 27 4
gpt4 key购买 nike

我有以下片段:

class Meta(type):
def __getattr__(self, name):
pass

class Klass(object):
__metaclass__ = Meta

def get(self, arg):
pass

现在,如果我这样做:

kls = Klass()
kls.get('arg')

一切都按预期工作(实例方法 get 被调用)。

但如果我这样做:

Klass.get('arg')

再次找到实例方法并给出异常,因为它被视为未绑定(bind)方法。

如何通过元类中定义的 __getattr__ 调用 Klass.get('arg')?我需要这个,因为我想将一个类上调用的所有方法代理到另一个对象(这将在 __getattr__ 中完成)。

最佳答案

您必须在类型上查找方法并手动传递第一个 (self) 参数:

type(Klass).get(Klass, 'arg')

这个问题正是special method names are looked up using this path 的原因;如果 Python 不这样做,自定义类本身将不可哈希或不可表示。

可以利用这个事实;而不是使用 get() 方法,使用 __getitem__,重载 [..] 索引语法,并让 Python 执行 type (ob).methodname(ob, *args) 为你跳舞:

class Meta(type):
def __getitem__(self, arg):
pass

class Klass(object):
__metaclass__ = Meta

def __getitem__(self, arg):
pass

然后 Klass()['arg']Klass['arg'] 按预期工作。

但是,如果您必须让 Klass.get() 有不同的行为(并且对此的查找被 Meta.__getattribute__ 拦截),您必须显式处理这在您的 Klass.get 方法中;如果在类上调用它,它将被少一个参数调用,您可以利用它并返回对类的调用:

_sentinel = object()

class Klass(object):
__metaclass__ = Meta

def get(self, arg=_sentinel):
if arg=_sentinel:
if isinstance(self, Klass):
raise TypeError("get() missing 1 required positional argument: 'arg'")
return type(Klass).get(Klass, self)
# handle the instance case ...

你也可以在 descriptor 中处理这个模仿方法对象:

class class_and_instance_method(object):
def __init__(self, func):
self.func = func
def __get__(self, instance, cls=None):
if instance is None:
# return the metaclass method, bound to the class
type_ = type(cls)
return getattr(type_, self.func.__name__).__get__(cls, type_)
return self.func.__get__(instance, cls)

并将其用作装饰器:

class Klass(object):
__metaclass__ = Meta

@class_and_instance_method
def get(self, arg):
pass

如果没有要绑定(bind)的实例,它会将查找重定向到元类:

>>> class Meta(type):
... def __getattr__(self, name):
... print 'Meta.{} look-up'.format(name)
... return lambda arg: arg
...
>>> class Klass(object):
... __metaclass__ = Meta
... @class_and_instance_method
... def get(self, arg):
... print 'Klass().get() called'
... return 'You requested {}'.format(arg)
...
>>> Klass().get('foo')
Klass().get() called
'You requested foo'
>>> Klass.get('foo')
Meta.get look-up
'foo'

可以在元类中应用装饰器:

class Meta(type):
def __new__(mcls, name, bases, body):
for name, value in body.iteritems():
if name in proxied_methods and callable(value):
body[name] = class_and_instance_method(value)
return super(Meta, mcls).__new__(mcls, name, bases, body)

然后您可以使用这个元类向类添加方法,而不必担心委托(delegate):

>>> proxied_methods = ('get',)
>>> class Meta(type):
... def __new__(mcls, name, bases, body):
... for name, value in body.iteritems():
... if name in proxied_methods and callable(value):
... body[name] = class_and_instance_method(value)
... return super(Meta, mcls).__new__(mcls, name, bases, body)
... def __getattr__(self, name):
... print 'Meta.{} look-up'.format(name)
... return lambda arg: arg
...
>>> class Klass(object):
... __metaclass__ = Meta
... def get(self, arg):
... print 'Klass().get() called'
... return 'You requested {}'.format(arg)
...
>>> Klass.get('foo')
Meta.get look-up
'foo'
>>> Klass().get('foo')
Klass().get() called
'You requested foo'

关于python - 使用类方法而不是同名的实例方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28699124/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com