如何获得对象的方法和属性的完整列表?


230
dir(re.compile(pattern)) 

不返回pattern作为列表的元素之一。即它返回:

['__copy__', '__deepcopy__', 'findall', 'finditer', 'match', 'scanner', 'search', 'split', 'sub', 'subn']

根据手册,它应该包含

对象的属性名称,其类的属性名称,以及递归其类的基类的属性。

它也说

该列表不一定完整。

有没有办法获得完整的清单?我一直以为dir返回一个完整的列表,但显然并没有...

另外:有没有办法只列出属性?还是只有方法?

编辑:这实际上是python中的错误->据称它已在3.0分支中修复(也许在2.6中也已修复)


5
使用dir()或检查模块通常是正确的方法。您re只是以模块为例还是要实现一个特殊的目标?

1
您确定该模式实际上在编译后仍保留为数据吗?我的印象是,编译模式的关键是产生解析给定模式所需的有限状态自动机。
Kyle Strand

@hop目录不能被类回避吗?例如,他们可以使他们在__dir__()
ytpillai

ytpillai:正确,但仅适用于Python3。即使如此,问题仍然是,这样的类是否属于“一般情况”下?

Answers:


140

对于属性的完整列表,简短的答案是:否。问题在于属性实际上定义为getattr内置函数接受的参数。由于用户可以重新实现__getattr__,突然允许任何类型的属性,因此没有可能的通用方法来生成该列表。该dir函数返回__dict__属性中的键,即,如果__getattr__未重新实现该方法,则所有可访问的属性。

对于第二个问题,这没有任何意义。实际上,方法是可调用的属性,仅此而已。但是,您可以过滤可调用的属性,并使用该inspect模块确定类的方法,方法或函数。


1
inpect.getmembers(re.compile(pattern))也不产生模式作为成员,因此它可能在内部使用dir ...这太糟了!
BartoszRadaczyński,

我还可以使用callable来检查它们是否是方法,但这不是重点...重点是我无法信任dir返回甚至实际上是公开可见的属性列表...
BartoszRadaczyński08年

2
inspect的意思是至少与dir()一样可信。另一方面,re是一个非常复杂的模块

您对“公开可见”的定义是什么?如果您的意思是“可以访问”,则由于给定的原因,这是一个丢失的原因。否则,“ dir”始终返回可通过getattr的默认实现访问的属性列表。
PierreBdR

2
dir(my_class)返回的内容与my_class .__ dict __。keys()不同。前者也输出类的方法,例如initdoc
JuanPi

58

这就是为什么__dir__()在python 2.6中添加了新方法的原因

看到:


我得到这个错误:>> DIR __(pyrenderdoc)回溯(最近通话最后一个):文件“<字符串>”,1号线,在<模块> NameError:名字'__dir '没有定义
莫纳塔拉巴尼

__dir__()是对象上的方法,不是函数-请阅读答案中的链接,这是
Moe

一线可打印所有属性及其值:pprint({k:getattr(ojb,k) for k in obj.__dir__()})
Czechnology

21

这是PierreBdR和Moe的答案的实用补充:

  • 对于Python> = 2.6和新型类dir()似乎就足够了。
  • 对于老式类,我们至少可以做一个标准模块来支持制表符补全:除了dir(),查找__class__,然后继续__bases__

    # code borrowed from the rlcompleter module
    # tested under Python 2.6 ( sys.version = '2.6.5 (r265:79063, Apr 16 2010, 13:09:56) \n[GCC 4.4.3]' )
    
    # or: from rlcompleter import get_class_members
    def get_class_members(klass):
        ret = dir(klass)
        if hasattr(klass,'__bases__'):
            for base in klass.__bases__:
                ret = ret + get_class_members(base)
        return ret
    
    
    def uniq( seq ): 
        """ the 'set()' way ( use dict when there's no set ) """
        return list(set(seq))
    
    
    def get_object_attrs( obj ):
        # code borrowed from the rlcompleter module ( see the code for Completer::attr_matches() )
        ret = dir( obj )
        ## if "__builtins__" in ret:
        ##    ret.remove("__builtins__")
    
        if hasattr( obj, '__class__'):
            ret.append('__class__')
            ret.extend( get_class_members(obj.__class__) )
    
            ret = uniq( ret )
    
        return ret

(测试代码和输出都被删除了简洁,但基本上对新风格的对象,我们似乎有相同的结果get_object_attrs()dir(),对于老式类主要除了dir()输出似乎是__class__属性。)


9

仅补充:

  1. dir()强大/最基本的工具。(最推荐
  2. 比其他解决方案dir()只是提供自己的方式处理的输出dir()

    是否列出第二级属性,重要的是您自己进行筛选,因为有时您可能希望筛选出带有前划线的内部变量__,但有时您很可能需要__doc__文档字符串。

  3. __dir__()dir()返回相同的内容。
  4. __dict__并且dir()是不同的。__dict__返回不完整的内容。
  5. 重要提示__dir__()有时作者可能出于某种目的用函数,值或类型覆盖它。

    这是一个例子:

    \\...\\torchfun.py in traverse(self, mod, search_attributes)
    445             if prefix in traversed_mod_names:
    446                 continue
    447             names = dir(m)
    448             for name in names:
    449                 obj = getattr(m,name)

    类型错误:描述__dir__'object'对象需要一个自变量

    PyTorch的作者将__dir__()方法修改为需要参数的东西。此修改使dir()失败。

  6. 如果要使用可靠的方案来遍历对象的所有属性,请记住,每个pythonic标准都可以被覆盖并且不成立,并且每个约定都可能不可靠。


5

这是我的操作方式,对于您要向其添加属性的简单自定义对象很有用:

给定使用obj = type("Obj",(object,),{})或简单创建的对象:

class Obj: pass
obj = Obj()

添加一些属性:

obj.name = 'gary'
obj.age = 32

然后,获取仅具有自定义属性的字典:

{key: value for key, value in obj.__dict__.items() if not key.startswith("__")}

# {'name': 'gary', 'age': 32}

Python 3.x中所有属性的列表:{key:键的值,如果不是,则为obj .__ dict __。items()中的值key.startswith(“ __”)} ['_ declared_fields']。keys()
above_c_level
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.