python装飾器(flaskフレームルーティング装飾器とdjangoソース装飾器、一般的な装飾器を列挙)5種類の装飾器があなたを連れて飛ぶ

3386 ワード

pythonアクセサリーは、既存の関数コードを変更することなく、いくつかの新しい機能の追加を実現することができ、pythonが開発した大神器です.
以下、私自身の学習経験と結びつけて、よくある装飾器を挙げて、参考にしてください.
0汎用装飾
特徴:
functools.wraps装飾パラメータfunc後、装飾器が生成した新しいtest関数は、独自の関数名とドキュメント情報を保持します.
パラメータは不定長パラメータとして設定し、必要に応じて自分で定義することもできます
import functools
def set_fun(func):
    @functools.wraps(func)
    def call_fun(*args,**kwargs):
        #        
        &"         "&
        return func(*args,**kwargs)
    return call_fun

@set_fun
def test(*args,**kwargs):
    pass

1入門デコレーション
単層関数実装装飾器
特徴:紹介、優美な装飾器の機能を実現して、構造は簡単ではっきりしていて、装飾器の中で被装飾関数を運行しないで、伝参の問題に関与しません
ミドルウェアのコードスタイルに似ています
def set_func(func):

    print(func.__name__)
    #        
    return func


@set_func
def index():
    print("hello,world")

index()

2進段装飾器
パラメータvalueを渡す二層関数装飾器
def route(value):

    def set_func(func):
        print(value)
        print(func.__name__)
        #        
        return func

    return set_func

@route("hhhhh") #         
def index():
    print("hello,world")

3 flask routeルーティング機能ルーティング装飾器ソースコード
flaskルーティング装飾器関数は、flask 0にある.1バージョンでFlaskクラスに定義されているインスタンスメソッド
例示的な方法は装飾器として定義され,複雑なパラメータを伝達する操作を実現するとともに参考に値する.

class Flask(object):
    ...

    def add_url_rule(self, rule, endpoint, **options):
        “”“       ”“”
        options['endpoint'] = endpoint
        options.setdefault('methods', ('GET',))
        self.url_map.add(Rule(rule, **options))

    def route(self, rule, **options):
        “”“       ”“”
        def decorator(f):
            self.add_url_rule(rule, f.__name__, **options)
            self.view_functions[f.__name__] = f
            return f
        return decorator

4 djangoユーザーがアクセサリーにログインしているかどうかを確認
ここに示すのはdjango 2です.0フレームワークのauthユーザーモジュールの装飾器で、ユーザーがログインしているかどうかを確認する機能です.
三層閉包関数は伝参と装飾機能を実現し,構造がより理解しやすく,参考に値する.
def user_passes_test(test_func, login_url=None, redirect_field_name=REDIRECT_FIELD_NAME):
    """
    Decorator for views that checks that the user passes the given test,
    redirecting to the log-in page if necessary. The test should be a callable
    that takes the user object and returns True if the user passes.
    """

    def decorator(view_func):
        @wraps(view_func)
        def _wrapped_view(request, *args, **kwargs):
            if test_func(request.user):
                return view_func(request, *args, **kwargs)
            path = request.build_absolute_uri()
            resolved_login_url = resolve_url(login_url or settings.LOGIN_URL)
            # If the login url is the same scheme and net location then just
            # use the path as the "next" url.
            login_scheme, login_netloc = urlparse(resolved_login_url)[:2]
            current_scheme, current_netloc = urlparse(path)[:2]
            if ((not login_scheme or login_scheme == current_scheme) and
                    (not login_netloc or login_netloc == current_netloc)):
                path = request.get_full_path()
            from django.contrib.auth.views import redirect_to_login
            return redirect_to_login(
                path, resolved_login_url, redirect_field_name)
        return _wrapped_view
    return decorator

end