241

クラスメソッドのデコレータにクラスフィールドを引数として渡すにはどうすればよいですか? 私がやりたいことは次のようなものです:

class Client(object):
    def __init__(self, url):
        self.url = url

    @check_authorization("some_attr", self.url)
    def get(self):
        do_work()

self.urlデコレータに渡すための self が存在しないと不平を言っています。これを回避する方法はありますか?

4

7 に答える 7

296

はい。クラス定義時にインスタンス属性を渡す代わりに、実行時にチェックします。

def check_authorization(f):
    def wrapper(*args):
        print args[0].url
        return f(*args)
    return wrapper

class Client(object):
    def __init__(self, url):
        self.url = url

    @check_authorization
    def get(self):
        print 'get'

>>> Client('http://www.google.com').get()
http://www.google.com
get

デコレーターはメソッドの引数をインターセプトします。最初の引数はインスタンスなので、そこから属性を読み取ります。属性名を文字列としてデコレータに渡し、属性名getattrをハードコーディングしたくない場合に使用できます。

def check_authorization(attribute):
    def _check_authorization(f):
        def wrapper(self, *args):
            print getattr(self, attribute)
            return f(self, *args)
        return wrapper
    return _check_authorization
于 2012-07-30T23:38:53.667 に答える
74

より簡潔な例は次のようになります。

#/usr/bin/env python3
from functools import wraps

def wrapper(method):
    @wraps(method)
    def _impl(self, *method_args, **method_kwargs):
        method_output = method(self, *method_args, **method_kwargs)
        return method_output + "!"
    return _impl

class Foo:
    @wrapper
    def bar(self, word):
        return word

f = Foo()
result = f.bar("kitty")
print(result)

どちらが印刷されますか:

kitty!
于 2016-04-29T18:13:02.693 に答える
5

できません。selfインスタンスが存在しないため、クラス本体にはありません。strたとえば、返された関数が実行できるインスタンスを検索する属性名を含むを渡すか、完全に別のメソッドを使用する必要があります。

于 2012-07-30T23:38:03.113 に答える