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

Bar*_*ski 217 python

dir(re.compile(pattern)) 
Run Code Online (Sandbox Code Playgroud)

不会将模式作为列表元素之一返回.即它返回:

['__copy__', '__deepcopy__', 'findall', 'finditer', 'match', 'scanner', 'search', 'split', 'sub', 'subn']
Run Code Online (Sandbox Code Playgroud)

根据手册,它应该包含

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

它也说

该清单未必完整.

有没有办法完成清单?我总是认为dir会返回一个完整的列表,但显然它没有......

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

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

Pie*_*BdR 136

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

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

  • inpect.getmembers(re.compile(pattern)) 也不产生作为成员的模式,所以它可能在内部使用 dir...这很糟糕! (2认同)
  • 检查意味着"至少和dir()一样"值得信赖.另一方面,re是一个非常复杂的模块 (2认同)
  • dir(my_class)返回与my_class .__ dict __.keys()不同的内容.前者还输出类的方法,如__init__和__doc__ (2认同)

Moe*_*Moe 55

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

看到:


ジョー*_*ョージ 18

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

- 对于Python> = 2.6 和新式类,dir()似乎就足够了;
- 对于旧式类,我们至少可以做标准模块支持标签完成的工作:除了dir(),寻找dir()- 然后去寻找它__class__:

# 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
Run Code Online (Sandbox Code Playgroud)

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


Chi*_*ron 7

仅补充:

  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)
    
    Run Code Online (Sandbox Code Playgroud)

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

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

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


小智 6

这就是我这样做的方式,对于您不断添加属性的简单自定义对象很有用:

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

class Obj: pass
obj = Obj()
Run Code Online (Sandbox Code Playgroud)

添加一些属性:

obj.name = 'gary'
obj.age = 32
Run Code Online (Sandbox Code Playgroud)

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

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

# {'name': 'gary', 'age': 32}
Run Code Online (Sandbox Code Playgroud)