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


120

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

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

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

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

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


あなたの例は「インスタンス変数」を示しています。クラス変数は別のものです。
S.Lott、2008

回答:


143

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

これを試して

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

7
FWIW、inspectモジュールは、すべてのインスタンスが持っているわけではないdictに依存するよりも信頼できる方法を提供します。
Thomas Wouters

1
どのようなインスタンスにdictはありませんか?私は一度も遭遇したことがありません。
カールマイヤー

3
intなどの特定の組み込み型。「x = 5」と言ってから「x .__ dict__」と言うと、AttributeErrorが表示されます
Eli Courtwright

6
また、スロットを使用するものすべて。
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構文では「プライベート」であるはずだからです。
マーティンW

3
@Martin:__methodプライベートで__method__あり、特別な方法であり、必ずしもプライベートではありません。特別なメソッドは、メソッドではなくオブジェクトの機能を定義するものです。
u0b34a0f6ae

2
__method__とても醜いですし、絶対に必要な場合を除いて、人々がそれらを使わないようにしようとするためにそのように名付けられたと思います。この場合、代替のvars()があります。
マーティンシャーバーン

私の場合、私はvars(ObjName)を呼び出してみましたが、指定されたクラス/オブジェクトのメソッドをキーとするディクショナリでdict_proxyを返しましたが、init要素のサインはありませんでした。なぜだと思いますか?
user228137

二重下線変数は__str____method__通常は言語自体用です。あなたはどうなりますstr(something)か?
Zizouz212

15

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

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


「インスタンス属性のリストを取得できない」ため、反対票を投じることは単純に間違っています。vars()とdictの両方が正確にそれを提供します。
カールマイヤー

2
私は彼が「すべての可能なインスタンス属性のリストを取得することはできない」と言っていたと思います。たとえば、最初にリクエストされたときに、インスタンス生成変数を追加するために、遅延生成と@propertyデコレーターをよく使用します。dictを使用すると、この変数が存在していても事前にではないことがわかります。
Eli Courtwright

5
私は反対票を投じなかったので、私が実際に言ったことに気づいてください。クラスのみ指定されたインスタンス属性のリストを取得することはできません。これは、質問に付随するコードが実行しようとしていることです。
Thomas Wouters

2
@Carl:誤ったコメントを書いたり、知識の欠如に基づいて反対票を投じる前に、自分自身を教育してください。
nikow 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

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('_')]
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.