Це може бути трохи пізно, але я прийняв рішення, використовуючи мета-класи Python (версія для декораторів також нижче).
Коли __init__
викликається під час виконання, він захоплює кожен з аргументів та їх значення та призначає їх як змінні екземпляри вашому класу. Таким чином, ви можете зробити структуру схожий клас без необхідності присвоювати кожне значення вручну.
У моєму прикладі немає перевірки помилок, тому його легше наслідувати.
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
Ось воно в дії.
>>> class MyClass(object):
__metaclass__ = MyStruct
def __init__(self, a, b, c):
pass
>>> my_instance = MyClass(1, 2, 3)
>>> my_instance.a
1
>>>
Я розмістив його на reddit та / u / matchu розмістив версію декоратора, яка чистіша. Я б радив вам використовувати його, якщо ви не хочете розширити версію метакласу.
>>> 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
>>>