Python中是否有内置方法来获取所有类的实例变量的数组?例如,如果我有这段代码:
class hi:
def __init__(self):
self.ii = "foo"
self.kk = "bar"
我有办法做到这一点:
>>> mystery_method(hi)
["ii", "kk"]
编辑:我最初错误地询问了类变量。
答案 0 :(得分:118)
每个对象都有一个__dict__
变量,其中包含所有变量及其值。
试试这个
>>> hi_obj = hi()
>>> hi_obj.__dict__.keys()
答案 1 :(得分:86)
使用vars()
class Foo(object):
def __init__(self):
self.a = 1
self.b = 2
vars(Foo()) #==> {'a': 1, 'b': 2}
vars(Foo()).keys() #==> ['a', 'b']
答案 2 :(得分:13)
您通常无法仅仅为类提供实例属性,至少在没有实例化类的情况下也是如此。但是,您可以获取给定实例的实例属性,或者给定类的类属性。请参阅“检查”模块。您无法获取实例属性列表,因为实例实际上可以包含任何属性,并且 - 在您的示例中 - 创建它们的常规方法是在__init__方法中分配它们。
例外情况是您的类使用插槽,插槽是类允许实例具有的固定属性列表。插槽在http://www.python.org/2.2.3/descrintro.html中有解释,但插槽有各种缺陷;它们会影响内存布局,因此多重继承可能会有问题,而且继承通常也需要考虑插槽。
答案 3 :(得分:13)
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
答案 4 :(得分:10)
您还可以测试对象是否具有特定变量:
>>> hi_obj = hi()
>>> hasattr(hi_obj, "some attribute")
答案 5 :(得分:6)
推荐
>>> print vars.__doc__
vars([object]) -> dictionary
Without arguments, equivalent to locals().
With an argument, equivalent to object.__dict__.
换句话说,它基本上只包装__dict__
答案 6 :(得分:5)
您的示例显示“实例变量”,而不是真正的类变量。
在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"
类变量由类的所有实例共享。
答案 7 :(得分: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'
答案 8 :(得分:0)
基于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
答案 9 :(得分:0)
有时您想基于公共/私有变量来过滤列表。例如
B
答案 10 :(得分:0)
您需要首先检查类,然后检查函数的字节码,然后复制字节码,最后, 使用 __code__.co_varnames
。 这很棘手,因为某些类使用 types
模块中的构造函数来创建它们的方法。我将在 GitHub 上提供代码。