如何找到Python对象的隐藏属性? (未出现在dir(obj)列表中的属性)

时间:2014-02-10 09:13:14

标签: python reflection

我正在使用IPython。

我声明了一个简单的类MyClass(object),并且在控制台中,当使用名称MyClass时,在dot运算符之后,我点击Tab

我得到的第一个建议之一是mro,意思是MyClass.mro。我点击了输入,我得到的输出是:

> <function mro>

现在我没有定义这个方法,当然它返回了我的类的方法解析顺序。

此方法未显示在dir(MyClass)返回的列表中,因此我的问题出现了:

如何找到类或其他对象的任何其他隐藏功能?

1 个答案:

答案 0 :(得分:4)

dir()并不试图提供完整的输出,只是在交互式解释器中有用的合理近似值。

Objects/object.c at line 1812下的源代码中可以找到输出中dir 包括__mro__mro的原因:

/* Helper for PyObject_Dir of type objects: returns __dict__ and __bases__.
   We deliberately don't suck up its __class__, as methods belonging to the
   metaclass would probably be more confusing than helpful.
*/
static PyObject *
_specialized_dir_type(PyObject *obj)

事实证明,mro方法和__mro__属性实际上是元类的属性,即type

>>> '__mro__' in object.__dict__
False
>>> '__mro__' in type.__dict__
True
>>> 'mro' in object.__dict__
False
>>> 'mro' in type.__dict__
True

因此它们没有显示在dir的输出中。这是否合理取决于。我相信大部分时间你真的想要看到元类定义的内容,因为它们是元类的内部。


dir()文档中所述,您可以自定义定义__dir__方法的输出。但是,这适用于类的实例

class A(object):
  def __dir__(self):
    return ['a', 'b', 'c']

print(dir(A()), dir(A))

输出:

(['a', 'b', 'c'], ['__class__', '__delattr__', '__dict__', '__dir__', '__doc__', '__format__', '__getattribute__', '__hash__', '__init__', '__module__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__'])

如果您还要自定义dir(A)的输出,则必须使用自定义元类:

class MyMeta(type):
  def __dir__(self):
    return ['a', 'b', 'c']


class A(object, metaclass=MyMeta):
  # __metaclass__ = MyMeta   # in python 2

  def __dir__(self):
    return ['a', 'b', 'c', 'd']


print(dir(A()), dir(A))

输出:

(['a', 'b', 'c', 'd'], ['a', 'b', 'c'])