파이썬 데코레이터에 추가 인수를 어떻게 전달합니까?
아래와 같은 데코레이터가 있습니다.
def myDecorator(test_func):
return callSomeWrapper(test_func)
def callSomeWrapper(test_func):
return test_func
@myDecorator
def someFunc():
print 'hello'
이 데코레이터를 향상시켜 아래와 같은 다른 주장을 받아들이고 싶습니다.
def myDecorator(test_func,logIt):
if logIt:
print "Calling Function: " + test_func.__name__
return callSomeWrapper(test_func)
@myDecorator(False)
def someFunc():
print 'Hello'
하지만이 코드는 오류를 제공합니다.
TypeError : myDecorator ()는 정확히 2 개의 인수 (1 개 제공)를받습니다.
함수가 자동으로 전달되지 않는 이유는 무엇입니까? 데코레이터 함수에 함수를 명시 적으로 전달하는 방법은 무엇입니까?
데코레이터를 함수처럼 호출하기 때문에 실제 데코레이터 인 다른 함수를 반환해야합니다.
def my_decorator(param):
def actual_decorator(func):
print("Decorating function {}, with parameter {}".format(func.__name__, param))
return function_wrapper(func) # assume we defined a wrapper somewhere
return actual_decorator
외부 함수에는 명시 적으로 전달한 모든 인수가 주어지며 내부 함수를 반환해야합니다. 내부 함수는 장식 할 함수를 전달하고 수정 된 함수를 반환합니다.
일반적으로 데코레이터가 래퍼 함수로 래핑하여 함수 동작을 변경하기를 원합니다. 다음은 함수가 호출 될 때 선택적으로 로깅을 추가하는 예입니다.
def log_decorator(log_enabled):
def actual_decorator(func):
@functools.wraps(func)
def wrapper(*args, **kwargs):
if log_enabled:
print("Calling Function: " + func.__name__)
return func(*args, **kwargs)
return wrapper
return actual_decorator
이 functools.wraps
호출은 이름 및 독 스트링과 같은 것을 래퍼 함수에 복사하여 원래 함수와 더 유사하게 만듭니다.
사용 예 :
>>> @log_decorator(True)
... def f(x):
... return x+1
...
>>> f(4)
Calling Function: f
5
다른 관점을 제공하기 위해 : 구문
@expr
def func(...): #stuff
다음과 같다
def func(...): #stuff
func = expr(func)
특히 expr
콜러 블로 평가되는 한 당신이 좋아하는 무엇이든 될 수 있습니다. 에서 특정 특히, expr
실내 장식 공장을 할 수 있습니다 : 당신은 몇 가지 매개 변수를주고 당신에게 장식을 제공합니다. 따라서 상황을 이해하는 더 좋은 방법은
dec = decorator_factory(*args)
@dec
def func(...):
다음으로 단축 할 수 있습니다.
@decorator_factory(*args)
def func(...):
물론 데코레이터 처럼 보이기 때문에 decorator_factory
사람들은 그것을 반영하기 위해 이름을 짓는 경향이 있습니다. 간접적 인 수준을 따르려고 할 때 혼란 스러울 수 있습니다.
데코레이터 인수를 선택적으로 만들 수있는 유용한 트릭을 추가하고 싶습니다. 또한 데코레이터를 재사용하고 중첩을 줄일 수 있습니다.
import functools
def myDecorator(test_func=None,logIt=None):
if not test_func:
return functools.partial(myDecorator, logIt=logIt)
@functools.wraps(test_func)
def f(*args, **kwargs):
if logIt==1:
print 'Logging level 1 for {}'.format(test_func.__name__)
if logIt==2:
print 'Logging level 2 for {}'.format(test_func.__name__)
return test_func(*args, **kwargs)
return f
#new decorator
myDecorator_2 = myDecorator(logIt=2)
@myDecorator(logIt=2)
def pow2(i):
return i**2
@myDecorator
def pow3(i):
return i**3
@myDecorator_2
def pow4(i):
return i**4
print pow2(2)
print pow3(2)
print pow4(2)
데코레이터를 수행하는 또 다른 방법입니다. 나는이 방법이 내 머리를 감싸는 가장 쉬운 방법임을 알았다.
class NiceDecorator:
def __init__(self, param_foo='a', param_bar='b'):
self.param_foo = param_foo
self.param_bar = param_bar
def __call__(self, func):
def my_logic(*args, **kwargs):
# whatever logic your decorator is supposed to implement goes in here
print('pre action baz')
print(self.param_bar)
# including the call to the decorated function (if you want to do that)
result = func(*args, **kwargs)
print('post action beep')
return result
return my_logic
# usage example from here on
@NiceDecorator(param_bar='baaar')
def example():
print('example yay')
example()
참고 URL : https://stackoverflow.com/questions/10176226/how-do-i-pass-extra-arguments-to-a-python-decorator
'development' 카테고리의 다른 글
R에서 사전 / 목록 작업 (0) | 2020.10.11 |
---|---|
WebResponse.GetResponseStream 반환을 문자열로 변환하는 방법은 무엇입니까? (0) | 2020.10.11 |
Linux bash / shell에서 이미지를 base64로 인코딩하는 방법 (0) | 2020.10.11 |
Ubuntu : OpenJDK 8-패키지를 찾을 수 없음 (0) | 2020.10.11 |
Android의 새로운 Google 로그인 (0) | 2020.10.11 |