Как вы программно устанавливаете атрибут?

204

Предположим , у меня есть объект питона xи строку s, как установить атрибут sна x? Так:

>>> x = SomeObject()
>>> attr = 'myAttr'
>>> # magic goes here
>>> x.myAttr
'magic'

Что за магия? Кстати, цель этого - кешировать вызовы x.__getattr__().

Ник
источник

Ответы:

289
setattr(x, attr, 'magic')

Для справки по этому вопросу:

>>> help(setattr)
Help on built-in function setattr in module __builtin__:

setattr(...)
    setattr(object, name, value)

    Set a named attribute on an object; setattr(x, 'y', v) is equivalent to
    ``x.y = v''.

Изменить: Тем не менее, вы должны отметить (как указано в комментарии), что вы не можете сделать это с «чистым» экземпляром object. Но, скорее всего, у вас есть простой подкласс объекта, где он будет работать нормально. Я настоятельно призываю ОП никогда не создавать экземпляры таких объектов.

Али Афшар
источник
12
Осторожно, однако, это не работает в вашем сценарии, когда вы создаете экземпляр объекта ().
S.Lott
3
Абсолютно верно, это не так. Я удобно проигнорировал это. Я настоятельно призываю ОП никогда не создавать экземпляры таких объектов.
Али Афшар
2
Черт, но это работает не во всех случаях, так как это было бы действительно полезно, например, для добавления dirtyатрибута к пользовательскому вводу ...
brice
1
@Brice: setattr работает практически во всех случаях. Из соображений эффективности и других причин «объект» запрограммирован так, что вы не можете добавить к нему дополнительные атрибуты. Вы можете сделать это с вашим собственным классом с __slots__атрибутом.
dirkjot
4
Это не работает, intа также. Вы можете объяснить, почему? (это на всех __builtin__?
Йенс Тиммерман
53

Обычно мы определяем классы для этого.

class XClass( object ):
   def __init__( self ):
       self.myAttr= None

x= XClass()
x.myAttr= 'magic'
x.myAttr

Тем не менее, вы можете, до некоторой степени, сделать это с setattrи getattrвстроенными функциями. Тем не менее, они не работают на случаяхobject напрямую.

>>> a= object()
>>> setattr( a, 'hi', 'mom' )
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: 'object' object has no attribute 'hi'

Они, однако, работают на всех видах простых классов.

class YClass( object ):
    pass

y= YClass()
setattr( y, 'myAttr', 'magic' )
y.myAttr
С. Лотт
источник
25
Любое понимание того, почему это не работает с экземплярами объекта ()?
Meawoppl
@meawoppl Вы должны задать это как новый вопрос
Тобиас Кинцлер
Спрашивается здесь .
Джаланб
1
я могу сделать это с модулями вместо классов?
Бонобо
21

пусть x будет объектом, тогда вы можете сделать это двумя способами

x.attr_name = s 
setattr(x, 'attr_name', s)
Виджей Шенкер
источник