Pythonでインスタンス変数を取得するにはどうすればよいですか?
-
01-07-2019 - |
質問
Python には、すべてのクラスのインスタンス変数の配列を取得する組み込みメソッドはありますか?たとえば、次のコードがあるとします。
class hi:
def __init__(self):
self.ii = "foo"
self.kk = "bar"
これを行う方法はありますか:
>>> mystery_method(hi)
["ii", "kk"]
編集:私はもともと誤ってクラス変数を要求していました。
解決
すべてのオブジェクトには __dict__
すべての変数とその値を含む変数。
これを試して
>>> hi_obj = hi()
>>> hi_obj.__dict__.keys()
他のヒント
vars() を使用する
class Foo(object):
def __init__(self):
self.a = 1
self.b = 2
vars(Foo()) #==> {'a': 1, 'b': 2}
vars(Foo()).keys() #==> ['a', 'b']
通常、クラスだけを指定した場合、少なくともクラスをインスタンス化しない限り、インスタンス属性を取得することはできません。ただし、インスタンスを指定してインスタンス属性を取得したり、クラスを指定してクラス属性を取得したりできます。「検査」モジュールを参照してください。インスタンス属性のリストを取得することはできません。インスタンスは実際には属性として何でも持つことができ、例のように、インスタンスを作成する通常の方法は __init__ メソッドでインスタンスに割り当てるだけです。
例外は、クラスがインスタンスに持つことを許可する属性の固定リストであるスロットを使用する場合です。スロットについては、 http://www.python.org/2.2.3/descrintro.html, しかし、スロットにはさまざまな落とし穴があります。これらはメモリ レイアウトに影響を与えるため、多重継承には問題が生じる可能性があり、一般に継承ではスロットも考慮する必要があります。
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
次のコマンドを使用して、オブジェクトに特定の変数があるかどうかをテストすることもできます。
>>> hi_obj = hi()
>>> hasattr(hi_obj, "some attribute")
提案する
>>> print vars.__doc__
vars([object]) -> dictionary
Without arguments, equivalent to locals().
With an argument, equivalent to object.__dict__.
言い換えれば、本質的には __dict__ をラップするだけです
あなたの例は実際にはクラス変数ではなく「インスタンス変数」を示しています。
覗く 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"
クラス変数は、クラスのすべてのインスタンスで共有されます。
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'