問題:route中的裝飾器為什么感覺和平時使用的不太一樣,裝飾器帶參數和不太參數有什么區別?被修飾的函數帶參數和不帶參數有什么區別?
測試1:裝飾器不帶參數,被修飾的函數也不帶參數。
def log(func): print"execute log" print func def use_log(): print "execute use log" def wrapper(): print "start" func() print "end" return return wrapper return use_log @log def cal(): print "1+2"
此時輸出為:
execute log <function cal at 0x7fa64535f668> #這里的function為cal的函數地址
如果執行cal()那么將會使用use_log函數,返回的是wrapper()
execute log <function cal at 0x7f42ee7a4668> execute use log
如果執行cal()的返回值,那么將執行cal()函數體的內容
result = cal() result()
結果為:
execute log <function cal at 0x7f38dc4d1668> execute use log start 1+2 end
測試2:如果裝飾器帶參數,被修飾的函數不帶參數
def log(func): #這里的func為裝飾器函數參數 print"execute log" print func #這里的func為裝飾器函數參數 def use_log(func): #這里的func為函數cal()的地址 print "execute use log" print func #這里的func為函數cal()的地址 def wrapper(): print "start" func() print "end" return return wrapper return use_log @log('log') def cal(): print "1+2" #這個時候數輸出結果為: execute log log execute use log <function cal at 0x7f0c666b46e0>
這個時候調用cal()那么將會執行wrapper()的函數體+cal()的函數體。
測試3:如果裝飾器不帶參數,被修飾的函數帶參數
def log(func): #func 為cal()函數的地址 print"execute log" def use_log(param): #param為cal的參數param print "execute use log" print param def wrapper(): print "start" func(param) #func 為cal()函數的地址,param為cal的參數param print "end" return return wrapper return use_log @log def cal(param): print "1+2" result = cal('cal') result() #執行的結果為: execute log execute use log cal start 1+2 end #如果注掉最后兩行代碼,那么只有輸出 execute log
測試4:如果裝飾器帶參數,被修飾的函數也帶參數。最復雜的情況。
def log(func): #func為裝飾器的參數 print"execute log" def use_log(func): #func為cal的函數地址 print "execute use log" print func #func為cal的函數地址 def wrapper(param): #param為cal的參數 print "start" func(param) print "end" return return wrapper return use_log @log('test') def cal(param): print "1+2" result = cal('cal') #執行的結果為: execute log execute use log <function cal at 0x7f23bbc6d6e0> start 1+2 end
經過上面的分析之后,再看flask中使用的是哪種情況:
樣例代碼:
from flask import Flask app = Flask(__name__) @app.route('/') def hello(): print 'execute hello function' return 'Hello, World!'
@app.route('/')的代碼如下:
def route(self, rule, **options): """A decorator that is used to register a view function for a given URL rule. This does the same thing as :meth:`add_url_rule` but is intended for decorator usage:: @app.route('/') def index(): return 'Hello World' For more information refer to :ref:`url-route-registrations`. :param rule: the URL rule as string :param endpoint: the endpoint for the registered URL rule. Flask itself assumes the name of the view function as endpoint :param options: the options to be forwarded to the underlying :class:`~werkzeug.routing.Rule` object. A change to Werkzeug is handling of method options. methods is a list of methods this rule should be limited to (``GET``, ``POST`` etc.). By default a rule just listens for ``GET`` (and implicitly ``HEAD``). Starting with Flask 0.6, ``OPTIONS`` is implicitly added and handled by the standard request handling. """ def decorator(f): endpoint = options.pop('endpoint', None) self.add_url_rule(rule, endpoint, f, **options) print "this param has been accessed" return f return decorator
可以看到裝飾器的參數為‘/’,被修飾的函數為:hello(),所以這里屬於第二種情況,即使不調用hello()函數,decorator的函數體也是被執行的,也就是說,只要使用裝飾器添加了路由規則,那么就會被加入到map中形成映射關系。