Jak programowo ustawić atrybut?

204

Załóżmy, że mam obiekt Pythona xi ciąg s, jak mogę ustawić atrybut sna x? Więc:

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

Jaka jest magia Nawiasem mówiąc, celem tego jest buforowanie połączeń do x.__getattr__().

Nacięcie
źródło

Odpowiedzi:

288
setattr(x, attr, 'magic')

Aby uzyskać pomoc na ten temat:

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

Edycja: Należy jednak pamiętać (jak wskazano w komentarzu), że nie można tego zrobić w przypadku „czystej” instancji object. Ale prawdopodobnie masz prostą podklasę obiektu, w której będzie dobrze działać. Usilnie nalegam na OP, aby nigdy nie tworzył takich obiektów.

Ali Afshar
źródło
12
Ostrożnie jednak nie działa to w twoim scenariuszu, w którym tworzysz instancję obiektu ().
S.Lott,
3
Absolutnie racja, nie ma. Dogodnie to zignorowałem. Usilnie nalegam na OP, aby nigdy nie tworzył takich obiektów.
Ali Afshar,
2
Cholera, wstyd, że nie działa we wszystkich przypadkach, ponieważ byłoby to naprawdę przydatne, na przykład, aby dodać dirtyatrybut do danych wejściowych użytkownika ...
Brice
1
@Brice: setattr działa w prawie wszystkich przypadkach. Ze względu na wydajność i inne powody programuje się „obiekt”, aby nie można było dodawać do niego dodatkowych atrybutów. Możesz to zrobić z własną klasą za pomocą __slots__atrybutu.
dirkjot
4
To też nie działa int. Czy możesz mi wytłumaczyć dlaczego? (czy to wszystko __builtin__?
Jens Timmerman
53

Zwykle definiujemy do tego klasy.

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

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

Jednakże, można do pewnego stopnia, to zrobić z setattri getattrwbudowanych funkcji. Nie działają one jednak objectbezpośrednio na instancje .

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

Działają jednak na wszelkiego rodzaju prostych klasach.

class YClass( object ):
    pass

y= YClass()
setattr( y, 'myAttr', 'magic' )
y.myAttr
S.Lott
źródło
25
Wgląd w to, dlaczego to nie działa z instancjami object ()?
meawoppl,
@meawoppl Powinieneś zadać to jako nowe pytanie
Tobias Kienzler
Zapytał tutaj .
jalanb
1
czy mogę to zrobić za pomocą modułów zamiast klas?
bonobo
21

niech x będzie obiektem, wtedy możesz to zrobić na dwa sposoby

x.attr_name = s 
setattr(x, 'attr_name', s)
Vijay Shanker
źródło