Potrebbe essere un po 'tardi, ma ho creato una soluzione usando le Meta-Classi di Python (anche la versione decoratore di seguito).
Quando __init__
viene chiamato durante il runtime, prende ciascuno degli argomenti e il loro valore e li assegna come variabili di istanza alla tua classe. In questo modo è possibile creare una classe simile alla struttura senza dover assegnare manualmente tutti i valori.
Il mio esempio non ha errori nel controllo, quindi è più facile da seguire.
class MyStruct(type):
def __call__(cls, *args, **kwargs):
names = cls.__init__.func_code.co_varnames[1:]
self = type.__call__(cls, *args, **kwargs)
for name, value in zip(names, args):
setattr(self , name, value)
for name, value in kwargs.iteritems():
setattr(self , name, value)
return self
Qui è in azione.
>>> class MyClass(object):
__metaclass__ = MyStruct
def __init__(self, a, b, c):
pass
>>> my_instance = MyClass(1, 2, 3)
>>> my_instance.a
1
>>>
L'ho pubblicato su reddit e / u / matchu ha pubblicato una versione decoratore che è più pulita. Ti incoraggio a usarlo a meno che tu non voglia espandere la versione della metaclasse.
>>> def init_all_args(fn):
@wraps(fn)
def wrapped_init(self, *args, **kwargs):
names = fn.func_code.co_varnames[1:]
for name, value in zip(names, args):
setattr(self, name, value)
for name, value in kwargs.iteritems():
setattr(self, name, value)
return wrapped_init
>>> class Test(object):
@init_all_args
def __init__(self, a, b):
pass
>>> a = Test(1, 2)
>>> a.a
1
>>>