web-dev-qa-db-ja.com

Pythonでインスタンス変数を取得する方法は?

Pythonにすべてのクラスのインスタンス変数の配列を取得するための組み込みメソッドはありますか?たとえば、このコードがある場合:

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

これを行う方法はありますか?

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

編集:私はもともとクラス変数を誤って要求していました。

104
Chris Bunch

すべてのオブジェクトには__dict__変数には、すべての変数とその値が含まれます。

これを試して

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

Vars()を使用します

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

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

通常、クラスをインスタンス化しない限り、クラスだけでインスタンス属性を取得することはできません。ただし、インスタンスが指定されたインスタンス属性、またはクラスが指定されたクラス属性を取得できます。 「検査」モジュールを参照してください。インスタンスは実際には属性として何でも持つことができるため、インスタンス属性のリストを取得することはできません。そして、あなたの例のように、それらを作成する通常の方法は__init__メソッドでそれらに単に割り当てることです。

例外は、クラスがスロットを使用する場合です。スロットは、クラスがインスタンスに許可する属性の固定リストです。スロットは http://www.python.org/2.2.3/descrintro.html で説明されていますが、スロットにはさまざまな落とし穴があります。これらはメモリレイアウトに影響するため、多重継承には問題が生じる可能性があり、一般に継承もスロットを考慮する必要があります。

14
Thomas Wouters

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
14
dmark

オブジェクトに特定の変数があるかどうかもテストできます:

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

提案する

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

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

つまり、本質的には単に__dict__をラップするだけです。

6
daniel

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'
5
tim.tadh

あなたの例は、実際のクラス変数ではなく、「インスタンス変数」を示しています。

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"

クラス変数は、クラスのすべてのインスタンスで共有されます。

5
S.Lott

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
Ethan Joffe