当前位置:  开发笔记 > 编程语言 > 正文

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

如何解决《如何在Python中获取实例变量?》经验,为你挑选了8个好方法。

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

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

有没有办法让我这样做:

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

编辑:我原本错误地要求类变量.



1> cnu..:

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

试试这个

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


FWIW,检查模块为您提供了比依赖__dict__更可靠的方法,并非所有实例都具备.
此外,任何使用__slots__的东西.
某些内置类型,例如int.尝试说"x = 5"然后"x .__ dict__",你会得到一个AttributeError
为什么要尝试获取int的类实例变量?它甚至不是一个阶级(尽管我可能错了).

2> Jeremy Cantr..:

使用vars()

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

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


+1我个人认为这种语法比使用__dict__更清晰,因为具有双下划线的属性在Python语法中应该是"私有的".
@Martin:`__ method`是私有的,`__ method__是一种特殊的方法,不一定是私有的; 我想说特殊方法定义了一个对象的功能而不是方法.
`__method__`是相当丑陋的,我认为他们已被命名为试图阻止人们使用它们,除非绝对必要.在这种情况下,我们有替代vars().

3> Thomas Woute..:

你通常不能只给出一个类给出的实例属性,至少在没有实例化类的情况下.但是,您可以获取给定实例的实例属性,或者给定类的类属性.请参阅"检查"模块.您无法获取实例属性列表,因为实例实际上可以包含任何属性,并且 - 在您的示例中 - 创建它们的常规方法是在__init__方法中分配它们.

例外情况是您的类使用插槽,插槽是类允许实例具有的固定属性列表.插槽在http://www.python.org/2.2.3/descrintro.html中有解释,但插槽有各种缺陷; 它们会影响内存布局,因此多重继承可能会有问题,并且继承通常也需要考虑插槽.


我没有downvote,请注意我实际说的内容:你不能得到一个实例属性列表*只给出一个类*,这就是问题附带的代码试图做的事情.
我认为他的意思是"你无法获得所有可能的实例属性列表".例如,我经常使用延迟生成和@property装饰器在第一次请求时添加实例变量.使用__dict__会告诉你这个变量一旦存在但不是事先.
@卡尔:请在写虚假评论并根据自己的知识不足而投票之前,对自己进行教育。

4> 小智..:

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


`exec('print object.%s \n \n')%i`应该写成`print getattr(object,i)`

5> daniel..:

您还可以测试对象是否具有特定变量:

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



6> daniel..:

建议

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

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

换句话说,它基本上只包装__dict__



7> S.Lott..:

您的示例显示"实例变量",而不是真正的类变量.

查找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"

类变量由类的所有实例共享.



8> tim.tadh..:

虽然不是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'

推荐阅读
喜生-Da
这个屌丝很懒,什么也没留下!
DevBox开发工具箱 | 专业的在线开发工具网站    京公网安备 11010802040832号  |  京ICP备19059560号-6
Copyright © 1998 - 2020 DevBox.CN. All Rights Reserved devBox.cn 开发工具箱 版权所有