如何在Python中获取实例变量?


120

Python中是否有内置方法来获取所有类的实例变量的数组?例如,如果我有以下代码:

class hi:
  def __init__(self):
    self.ii = "foo"
    self.kk = "bar"

有没有办法让我做到这一点:

>>> mystery_method(hi)
["ii", "kk"]

编辑:我最初是错误地要求类变量。


您的示例显示“实例变量”。类变量是另一回事。
S.Lott

Answers:


143

每个对象都有一个__dict__变量,其中包含所有变量及其值。

试试这个

>>> hi_obj = hi()
>>> hi_obj.__dict__.keys()

7
FWIW,检查模块为您提供了一种比依赖于dict更为可靠的方法,并非所有实例都具有dict
Thomas Wouters

1
什么样的实例没有字典?我从未遇到过。
卡尔·梅耶

3
某些内置类型,例如int。尝试说“ x = 5”,然后说“ x .__ dict__”,您会得到AttributeError
Eli Courtwright

6
同样,任何使用slot的东西。
Thomas Wouters

2
您为什么要尝试获取int的类实例变量?它甚至都不是一堂课(尽管我可能错了)。
Martin Sherburn

103

使用vars()

class Foo(object):
    def __init__(self):
        self.a = 1
        self.b = 2

vars(Foo()) #==> {'a': 1, 'b': 2}
vars(Foo()).keys() #==> ['a', 'b']

6
+1我个人认为此语法比使用dict更干净,因为带有双下划线的属性在Python语法中应该是“私有”的。
Martin W

3
@Martin:__method是私有的,__method__是一种特殊的方法,不一定是私有的;我想说的是,特殊方法定义对象的功能而不是方法。
u0b34a0f6ae 2009年

2
__method__非常丑陋,除非绝对必要,否则我认为它们已被命名为试图阻止人们使用它们的方式。在这种情况下,我们可以选择vars()。
Martin Sherburn

在我的情况下,我尝试调用vars(ObjName),它返回带有字典的dict_proxy,该字典的键是给定类/对象的方法,但没有init元素的迹象。任何猜测为什么?
user228137 2014年

Double Underscore Variables __str____method__通常用于语言本身。当你str(something)怎么办?
Zizouz212

15

通常,仅给定一个类就不能获得实例属性,至少不能不实例化该类。但是,您可以获取给定实例的实例属性,也可以获取给定类的类属性。请参阅“检查”模块。您无法获得实例属性的列表,因为实例实际上可以将任何东西作为属性,而且-如您的示例中所示-创建它们的通常方法是只在__init__方法中对其进行分配。

例外是您的类使用插槽,插槽是类允许实例具有的固定属性列表。插槽在http://www.python.org/2.2.3/descrintro.html中进行了说明,但是插槽存在各种陷阱。它们会影响内存布局,因此多重继承可能会出现问题,并且一般而言,继承也必须考虑插槽。


拒绝投票是因为“您无法获得实例属性列表”完全是错误的:vars()和dict都可以为您提供准确的信息。
卡尔·梅耶

2
我认为他的意思是“您无法获得所有可能的实例属性的列表”。例如,我经常使用延迟生成和@property装饰器在首次请求实例变量时添加它。使用dict可以告诉您有关该变量的信息,但该变量不存在。
伊莱·考特赖特'

5
我没有投票,请注意我实际上在说什么:仅给出一个类,您就无法获得实例属性的列表,这是该问题随附的代码尝试执行的操作。
Thomas Wouters

2
@卡尔:请在写虚假评论并因您缺乏知识而投下低估之前,对自己进行教育。
尼科夫,2009年

14

Vars()和dict方法都将适用于OP发布的示例,但不适用于“松散”定义的对象,例如:

class foo:
  a = 'foo'
  b = 'bar'

要打印所有不可调用的属性,可以使用以下功能:

def printVars(object):
    for i in [v for v in dir(object) if not callable(getattr(object,v))]:
        print '\n%s:' % i
        exec('print object.%s\n\n') % i

5
exec('print object.%s\n\n') % i应该写为print getattr(object, i)
user102008 2011年

11

您还可以使用以下方法测试对象是否具有特定变量:

>>> hi_obj = hi()
>>> hasattr(hi_obj, "some attribute")

6

您的示例显示了“实例变量”,而不是真正的类变量。

查找hi_obj.__class__.__dict__.items()类变量,以及其他其他类成员,例如成员函数和包含的模块。

class Hi( object ):
    class_var = ( 23, 'skidoo' ) # class variable
    def __init__( self ):
        self.ii = "foo" # instance variable
        self.jj = "bar"

类变量由该类的所有实例共享。


6

建议

>>> print vars.__doc__
vars([object]) -> dictionary

Without arguments, equivalent to locals().
With an argument, equivalent to object.__dict__.

换句话说,它实际上只是包装__dict__


5

尽管不是直接回答OP问题,但是有一种很不错的方法可以找出函数范围内的变量。看一下这段代码:

>>> def f(x, y):
    z = x**2 + y**2
    sqrt_z = z**.5
    return sqrt_z

>>> f.func_code.co_varnames
('x', 'y', 'z', 'sqrt_z')
>>> 

func_code属性中包含各种有趣的东西。它可以让您做一些很酷的事情。这是我如何使用此示例:

def exec_command(self, cmd, msg, sig):

    def message(msg):
        a = self.link.process(self.link.recieved_message(msg))
        self.exec_command(*a)

    def error(msg):
        self.printer.printInfo(msg)

    def set_usrlist(msg):
        self.client.connected_users = msg

    def chatmessage(msg):
        self.printer.printInfo(msg)

    if not locals().has_key(cmd): return
    cmd = locals()[cmd]

    try:
        if 'sig' in cmd.func_code.co_varnames and \
                       'msg' in cmd.func_code.co_varnames: 
            cmd(msg, sig)
        elif 'msg' in cmd.func_code.co_varnames: 
            cmd(msg)
        else:
            cmd()
    except Exception, e:
        print '\n-----------ERROR-----------'
        print 'error: ', e
        print 'Error proccessing: ', cmd.__name__
        print 'Message: ', msg
        print 'Sig: ', sig
        print '-----------ERROR-----------\n'

2

建立在dmark的答案上以获取以下内容,如果您希望获得sprintf的等效功能,这将很有用,并希望能对某人有所帮助...

def sprint(object):
    result = ''
    for i in [v for v in dir(object) if not callable(getattr(object, v)) and v[0] != '_']:
        result += '\n%s:' % i + str(getattr(object, i, ''))
    return result

0

有时您想根据公共/私有变量来过滤列表。例如

def pub_vars(self):
    """Gives the variable names of our instance we want to expose
    """
    return [k for k in vars(self) if not k.startswith('_')]
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.