Как получить переменные экземпляра в Python?

Есть ли встроенный метод в Python для получения массива всех переменных экземпляра класса? Например, если у меня есть этот код:

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

Есть ли способ для меня сделать это:

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

Изменение: я изначально ошибочно попросил переменные класса.

Ответ 1

Каждый объект имеет переменную __dict__, содержащую все переменные и ее значения в ней.

Попробуйте это

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

Ответ 2

Использовать vars()

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

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

Ответ 3

Оба метода 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

Обычно вы не можете получить атрибуты экземпляра, заданные только классом, по крайней мере, не без создания экземпляра класса. Вы можете получить атрибуты экземпляра с учетом экземпляра или атрибутов класса, заданных классом. См. Модуль "проверка". Вы не можете получить список атрибутов экземпляра, потому что экземпляры действительно могут иметь что-либо как атрибут, и - как в вашем примере - обычный способ их создания - просто назначить им метод __init__.

Исключением является то, что ваш класс использует слоты, который является фиксированным списком атрибутов, которые класс позволяет экземплярам иметь. Слоты объясняются в http://www.python.org/2.2.3/descrintro.html, но есть различные подводные камни с слотами; они влияют на макет памяти, поэтому множественное наследование может быть проблематичным, и наследование вообще должно учитывать слоты.

Ответ 5

Вы также можете проверить, имеет ли объект конкретную переменную с:

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

Ответ 6

Предложите

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

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

В других словах он по существу просто обертывает __dict__

Ответ 7

В вашем примере показаны "переменные экземпляра", а не действительно переменные класса.

Посмотрите 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"

Переменные класса разделяются всеми экземплярами класса.

Ответ 8

Хотя это и не является прямым ответом на вопрос ОП, есть довольно приятный способ узнать, какие переменные находятся в области действия функции. взгляните на этот код:

>>> 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'

Ответ 9

построен на dmark answer, чтобы получить следующее, что полезно, если вы хотите получить эквивалент 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