To może być trochę za późno, ale zrobiłem rozwiązanie za pomocą Python Meta-Classes (poniżej również wersja dekoratora).
Kiedy __init__
jest wywoływany w czasie wykonywania, pobiera każdy z argumentów i ich wartość i przypisuje je jako zmienne instancji do twojej klasy. W ten sposób możesz stworzyć klasę podobną do struktury bez konieczności przypisywania każdej wartości ręcznie.
W moim przykładzie nie ma sprawdzania błędów, więc łatwiej jest naśladować.
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
Tutaj jest w akcji.
>>> class MyClass(object):
__metaclass__ = MyStruct
def __init__(self, a, b, c):
pass
>>> my_instance = MyClass(1, 2, 3)
>>> my_instance.a
1
>>>
I napisali go na reddit i / u / matchu napisali wersję dekorator, który jest czystsze. Zachęcam do korzystania z niego, chyba że chcesz rozszerzyć wersję metaklasy.
>>> 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
>>>