42. 데코레이터 decorator

Joy·2020년 4월 6일
1

클래스에서 메서드를 만들 때 @staticmethod, @classmethod, @abstractmethod 등을 붙였는데, 이렇게 @로 시작하는 것들이 데코레이터. - 함수(메서드)를 장식

함수를 수정하지 않은 상태에서 추가 기능을 구현할 때 사용
예)

def hello():
    print('hello 함수 시작')
    print('hello')
    print('hello 함수 끝')
 
def world():
    print('world 함수 시작')
    print('world')
    print('world 함수 끝')
 
hello()
world()

결과:
hello 함수 시작
hello
hello 함수 끝
world 함수 시작
world
world 함수 끝

다른 함수에도 적용할 때 번거로움 -> 데코레이터 사용

def trace(func):                             # 호출할 함수를 매개변수로 받음
    def wrapper():                           # 호출할 함수를 감싸는 함수
        print(func.__name__, '함수 시작')    # __name__으로 함수 이름 출력
        func()                               # 매개변수로 받은 함수를 호출
        print(func.__name__, '함수 끝')
    return wrapper                           # wrapper 함수 반환
 
def hello():
    print('hello')
 
def world():
    print('world')
 
trace_hello = trace(hello)    # 데코레이터에 호출할 함수를 넣음
trace_hello()                 # 반환된 함수를 호출
trace_world = trace(world)    # 데코레이터에 호출할 함수를 넣음
trace_world()                 # 반환된 함수를 호출

result:
hello 함수 시작
hello
hello 함수 끝
world 함수 시작
world
world 함수 끝

데코레이터 trace는 호출할 함수를 매개변수로 받음
trace 함수 안에서는 호출할 함수를 감싸는 함수 wrappe
wrapper 함수에서는 함수의 시작을 알리는 문자열을 출력하고, trace에서 매개변수로 받은 func를 호출, 함수의 끝을 알리는 문자열을 출력
매개변수로 받은 함수의 원래 이름을 출력할 때는 name 속성을 활용
wrapper 함수를 다 만들었으면 return을 사용하여 wrapper 함수 자체를 반환
-> 함수 안에서 함수를 만들고 반환하는 클로저
데코레이터를 사용할 때는 trace에 호출할 함수 hello 또는 world를 넣습니다. 그다음에 데코레이터에서 반환된 함수를 호출

@로 데코레이터 사용

@데코레이터1
@데코레이터2
def 함수이름():
    코드

예)
hello와 world 함수 위에 @trace를 붙인 뒤에 hello와 world 함수를 그대로 호출하면 끝

def trace(func):                             # 호출할 함수를 매개변수로 받음
    def wrapper():
        print(func.__name__, '함수 시작')    # __name__으로 함수 이름 출력
        func()                               # 매개변수로 받은 함수를 호출
        print(func.__name__, '함수 끝')
    return wrapper                           # wrapper 함수 반환
 
@trace    # @데코레이터
def hello():
    print('hello')
 
@trace    # @데코레이터
def world():
    print('world')
 
hello()    # 함수를 그대로 호출
world()    # 함수를 그대로 호출

매개변수와 반환값 처리하는 데코레이터

예)

def trace(func):          # 호출할 함수를 매개변수로 받음
    def wrapper(a, b):    # 호출할 함수 add(a, b)의 매개변수와 똑같이 지정
        r = func(a, b)    # func에 매개변수 a, b를 넣어서 호출하고 반환값을 변수에 저장
        print('{0}(a={1}, b={2}) -> {3}'.format(func.__name__, a, b, r))  # 매개변수와 반환값 출력
        return r          # func의 반환값을 반환
    return wrapper        # wrapper 함수 반환
 
@trace              # @데코레이터
def add(a, b):      # 매개변수는 두 개
    return a + b    # 매개변수 두 개를 더해서 반환
 
print(add(10, 20))

매개변수가 고정되지 않고으면? - wrapper를 가변인수함수로

def trace(func):                     # 호출할 함수를 매개변수로 받음
    def wrapper(*args, **kwargs):    # 가변 인수 함수로 만듦
        r = func(*args, **kwargs)    # func에 args, kwargs를 언패킹하여 넣어줌
        print('{0}(args={1}, kwargs={2}) -> {3}'.format(func.__name__, args, kwargs, r))
                                     # 매개변수와 반환값 출력
        return r                     # func의 반환값을 반환
    return wrapper                   # wrapper 함수 반환

매개변수가 있는 데코레이터

@데코레이터(인수)
def 함수이름():
    코드
    def real_decorator(func):    # 호출할 함수를 매개변수로 받음
        def wrapper(a, b):       # 호출할 함수의 매개변수와 똑같이 지정
            r = func(a, b)       # func를 호출하고 반환값을 변수에 저장
            if r % x == 0:       # func의 반환값이 x의 배수인지 확인
                print('{0}의 반환값은 {1}의 배수입니다.'.format(func.__name__, x))
            else:
                print('{0}의 반환값은 {1}의 배수가 아닙니다.'.format(func.__name__, x))
            return r             # func의 반환값을 반환
        return wrapper           # wrapper 함수 반환
    return real_decorator        # real_decorator 함수 반환
 
@is_multiple(3)     # @데코레이터(인수)
def add(a, b):
    return a + b
 
print(add(10, 20))
print(add(2, 5))

클래스로 데코레이터 만들기

call 메서드 필요 : 인스턴스를 함수처럼 호출

class Trace:
    def __init__(self, func):    # 호출할 함수를 인스턴스의 초깃값으로 받음
        self.func = func         # 호출할 함수를 속성 func에 저장
 
    def __call__(self):
        print(self.func.__name__, '함수 시작')    # __name__으로 함수 이름 출력
        self.func()                               # 속성 func에 저장된 함수를 호출
        print(self.func.__name__, '함수 끝')
 
@Trace    # @데코레이터
def hello():
    print('hello')
 
hello()    # 함수를 그대로 호출

보통 데코레이터는 프로그램의 버그를 찾는 디버깅, 함수의 성능 측정, 함수 실행 전에 데이터 확인 등에 활용

핵심

ref: https://dojang.io/mod/page/view.php?id=2454

문제 : HTML 태그 데코레이터 만들기

표준 입력으로 HTML 태그 이름 두 개가 입력됩니다. 다음 소스 코드에서 함수의 반환값을 HTML 태그로 감싸는 데코레이터를 만드세요. HTML 태그는 웹 페이지에 사용하는 문법이며 문자열,

문자열

처럼 <태그이름>으로 시작하며 </태그이름>으로 끝납니다.

def html_tag(tag):
    def real_decorator(func):
        def wrapper():
            return '<{0}>{1}</{0}>'.format(tag, func())
        return wrapper
    return real_decorator

html_tag에는 입력값 a와 b를 넣고 있으므로 매개변수가 있는 데코레이터
tml_tag 함수 안에서는 실제 데코레이터 역할을 하는 real_decorator def real_decorator(func):와 같이 호출할 함수를 매개변수로
함수 안에서는 다시 wrapper 함수 - 문자열을 다 만들었으면 return으로 결과를 반환

profile
roundy

0개의 댓글