我有一些函数,在正常情况下,这些函数是使用用户输入提供的参数调用的。但是,使用某些系列参数调用其中一些函数是有效的,这些参数是在运行时根据某些系统状态确定的。
我希望用户能够有选择地指示我的程序使用所有有效输入调用这些函数并返回每次调用的结果。我认为一个装饰器可以像激活开关一样工作,用于具有另一个装饰器的函数,该装饰器指示要使用的参数系列会很好地工作。
此外,我需要保留函数签名和元数据。这对我的程序的运行至关重要。
这是我尝试过的,但它不起作用。它基于这个例子。
>>> from decorator import decorator
>>> def the_list():
... return ["foo", "bar", "baz"]
...
>>> import itertools
>>> @decorator
... def do_all(func):
... # this will do nothing (besides wrap in a tuple) unless func is decorated with @gets_arg_from
... if hasattr(func, 'get_from'):
... return tuple(func(*args) for args in itertools.product(*(list_fun() for list_fun in func.get_from)))
... else:
... return (func(),)
...
>>> def gets_arg_from(*list_funcs):
... # this will do nothing to func unless decorated with @do_all
... def gets_arg_from(func, *args, **kwargs):
... func.get_from = list_funcs
... return func(*args, **kwargs)
... return decorator(gets_arg_from)
...
>>> @gets_arg_from(the_list)
... def print_func(word):
... # this will print its argument unless decorated with @do_all
... # at that point it will print every element returned by the_list()
... print word
...
>>> print_func("foo")
foo
>>> all = decorator(do_all, print_func)
>>> all()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: print_func() takes exactly 1 argument (0 given)
>>> print_func.get_from
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
AttributeError: 'function' object has no attribute 'get_from'
我的预期是:
>>> all()
("foo", "bar", "baz")
我注意到的是错误的:
gets_arg_from
不将get_from
属性添加到func
.- 关于我使用该符号的某些事情
@gets_arg_from(the_list)
是错误的。它认为我正在尝试传递两个参数(但为什么这会成为问题呢?)
至于我的动机,我想到了装饰器,因为实际上有数百个这样的例程,它们的实现细节(以及它们的功能正确性)经常变化,我不想用它inspect
来推理做什么基于它们的参数名称,我也不想do_all
对每个有意义的函数的功能进行硬编码。类方法可能有效,但就我的目的而言,它们是语义设计的。此外,为了其他可能需要维护我的代码的人,我认为让他们应用装饰器而不用担心其余部分比使用某些参数名称或将函数放在某个类中或任何。我意识到这个问题可能听起来很奇怪,所以我认为这个脚注可能会让我看起来不那么像个疯子。