- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我有以下片段:
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/
我写了我遇到的问题,但现在我展示了所有代码xml文件 it can't contain other elements it can contain some other elements
我有两个 Controller using Microsoft.AspNetCore.Authorization; using Microsoft.AspNetCore.Hosting; using
我正在尝试保存几个同名的文件。我想做一些名字做这样的事情:file.extension file[1].extension file[2].extension 我试过这个 http://www.nas
我正在尝试创建一个脚本,该脚本将在多个页面上单击相同名称的按钮,但这些按钮具有不同的属性。有些有 id、一些名称、一些值和一些数据作为标识符。这个函数应该有一个属性列表,当我运行脚本时,它应该进入该列
从 Oracle 迁移我想重用一些定义,例如: ALTER TABLE CLIENTUSERS ADD CONSTRAINT UK_CLIENTUSERS_CLIENTUS UNIQUE (CLIEN
我有一个关于 unix 命令行的问题。我有很多这样的文件: /f/f1/file.txt /f/f2/file.txt /f/f3/file.txt and so on. 我想将所有 file.txt
我在表单“delsel2”中有几个复选框(全部具有相同的名称),如下所示: 我希望能够在选择其中任何一个时简单地发出警报,并显示一条消息指示是否选中其中任何一个。 我已经想出了这个,但它不起作用
假设您在子目录 project_1, ..., project_10 中有 10 个项目,它们都在父目录 projects 中。 所有项目都使用相同的库,在 project/project_n/lib
我有一个生成 iOS 框架的 Xcode 项目。我有两个版本的二进制文件要生成 - 一个供内部使用,没有许可要求,另一个供外部使用,但有许可要求。 我想我会为此制定两个目标,并使用不同的 swift
我已经读过有关 jQuery 验证的内容,但我仍然坚持在表单中需要控制数组。 我有这个代码: //already link the validation js script of course
我对这个领域很陌生。我正在尝试阅读一个示例程序。 第一个是team.c #include "support.h" struct team_t team = { "", /* first membe
我有一个包含一组复选框的 HTML 表单: Apple Banana Strawberry 当我单击一个按钮时,我希望将所有选中的同名复选框放在一个数组中。 因此,如果用户选择 Apple 和
我将第一个表存储在数据库中,如图所示,我必须对其应用查询以像第二个表一样显示它 最佳答案 这是使用条件聚合的一个选项: select subject, max(case when exa
对于我文件夹中的每个 mp3 文件,我都有一个同名的 jpg 文件,我试图用 cmd 实现的是将该名称变成一个没有文件扩展名的变量名 这是我只使用 mp3 名称的代码 for %%a in ("*.m
我有 2 个数据库....a 和 b 我在这两个数据库中都有表“t”。 现在我正在从数据库“b”中删除表 t。 我创建了一个 View “t”(请参阅 View 名称与已删除的表相同) 数据库“b
我正在创建一个有特定问题的 SCSS 网格 - 我想使用一个变量名,例如 $pad(用于填充值),但是 $pad变量在不同的媒体断点中需要不同。 首先通过动态创建断点并在其中设置 $pad 值的 mi
一些背景知识: 我正在尝试将我在 Solaris 上运行的一些 .ksh 文件移植到使用 Cygwin 的 Windows 上运行。在 Solaris 机器上运行的 ksh 实现与可在处理子 shel
当执行 tar 提取操作时,有时内容会直接提取到父文件夹,这很糟糕,因为它们变得杂乱无章。例如: tar -xzf foo1.tar.gz 摘录: ./file1 ./file2 ./file3 解决
我想做这样的事情: class X: @classmethod def id(cls): return cls.__name__ def id(self):
我正在看 John De Goes “FP to the Max”视频。在代码中,他做了这样的事情来获取隐式对象: object Program { def apply[F[_]](imp
我是一名优秀的程序员,十分优秀!