Python에서 인스턴스 변수를 얻는 방법은 무엇입니까?


120

모든 클래스의 인스턴스 변수 배열을 가져 오는 내장 메소드가 Python에 있습니까? 예를 들어이 코드가있는 경우 :

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

이렇게 할 수있는 방법이 있습니까?

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

편집 : 원래 클래스 변수를 잘못 요청했습니다.


귀하의 예는 "인스턴스 변수"를 보여줍니다. 클래스 변수는 또 다른 것입니다.
S.Lott

답변:


143

모든 개체에는 __dict__모든 변수와 그 값을 포함 하는 변수가 있습니다.

이 시도

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

7
FWIW, inspect 모듈은 모든 인스턴스에있는 것은 아니지만 dict에 의존하는 것보다 더 안정적인 방법을 제공합니다 .
Thomas Wouters

1
어떤 종류의 인스턴스에 dict가 없습니까? 나는 한 번도 만난 적이 없습니다.
Carl Meyer

3
int와 같은 특정 내장 유형. "x = 5"와 "x .__ dict__"를 차례로 말하면 AttributeError가 발생합니다
Eli

6
또한 slots 을 사용하는 모든 것 .
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를 사용하는 것보다 더 깨끗하다고 ​​생각합니다. 이중 밑줄이있는 속성은 파이썬 구문에서 "비공개"로 간주되기 때문입니다.
Martin W

3
@Martin : __method비공개이며, __method__반드시 비공개가 아닌 특별한 방법입니다. 특별한 방법이 방법보다는 객체의 기능을 정의한다고 말하고 싶습니다.
u0b34a0f6ae

2
__method__아주 추하고, 절대적으로 필요한 경우가 아니라면 사람들이 사용하지 못하도록 막기 위해 이름이 붙여진 것 같습니다. 이 경우 대체 vars ()가 있습니다.
Martin Sherburn

제 경우에는 vars (ObjName) 호출을 시도했고 키가 주어진 클래스 / 객체의 메서드 인 딕셔너리와 함께 dict_proxy를 반환하지만 init 요소의 부호는 없습니다 . 그 이유는 무엇입니까?
user228137

이중 밑줄 변수는 __str__, __method__일반적으로 언어 자체에 대한 것입니다. 당신은 어떻게됩니까 str(something)?
Zizouz212

15

일반적으로 클래스를 인스턴스화하지 않고는 클래스에만 주어진 인스턴스 속성을 가져올 수 없습니다. 하지만 주어진 인스턴스 속성이나 클래스가 주어진 클래스 속성을 얻을 수 있습니다. '검사'모듈을 참조하십시오. 인스턴스는 실제로 어떤 것도 속성으로 가질 수 있기 때문에 인스턴스 속성 목록을 가져올 수 없으며, 예제에서와 같이 일반적인 방법은 __init__ 메서드에서 할당하는 것입니다.

클래스가 인스턴스를 가질 수 있도록 허용하는 고정 된 속성 목록 인 슬롯을 사용하는 경우는 예외입니다. 슬롯은 http://www.python.org/2.2.3/descrintro.html 에 설명되어 있지만 슬롯에는 여러 가지 함정이 있습니다. 메모리 레이아웃에 영향을 미치므로 다중 상속이 문제가 될 수 있으며 일반적으로 상속도 슬롯을 고려해야합니다.


"인스턴스 속성 목록을 얻을 수 없습니다"라는 이유로 반대표를 던지는 것은 단순히 잘못되었습니다. vars ()와 dict 모두 정확하게 제공합니다.
Carl Meyer

2
나는 그가 "가능한 모든 인스턴스 속성의 목록을 얻을 수 없다"는 뜻이라고 생각한다. 예를 들어, 처음 요청할 때 인스턴스 변수를 추가하기 위해 종종 지연 생성과 @property 데코레이터를 사용합니다. dict를 사용하면 이 변수가 존재하면 알 수 있지만 사전에는 알 수 없습니다.
Eli Courtwright

5
나는 반대표를 던지지 않았고 내가 실제로 말한 것을 주목하십시오 : 질문에 수반되는 코드가 시도하는 것 인 class 만 주어진 인스턴스 속성 목록을 얻을 수 없습니다 .
Thomas Wouters

2
@Carl : 허위 댓글을 작성하고 지식이 부족하여 투표하기 전에 자신을 교육하십시오.
nikow

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.