国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Python > 正文

Python中裝飾器兼容加括號和不加括號的寫法詳解

2020-02-16 01:47:26
字體:
供稿:網(wǎng)友

使用Django的時候,我發(fā)現(xiàn)一個很神奇的裝飾器: @login_required, 這是控制一個view的權(quán)限的,比如一個視圖必須登錄才可以訪問,可以這樣用:

@login_requireddef my_view(request): ... return render(...)

同時,如果要達(dá)到這樣一種效果:如果用戶沒有登錄,那么就把用戶重定向到登錄界面,可以這樣用:

@login_required(login_url='/accounts/login/')def my_view(request): ... return render(...)

所以這個裝飾器可以帶括號寫,又可以不帶括號寫。很神奇有沒有。正常的接收參數(shù)的裝飾器,就算沒參數(shù)也應(yīng)該寫成@login_required的

好奇去查了一下,在stackoverflow找到一種實現(xiàn),挺有意思的。先曬出答案:

def doublewrap(f): ''' a decorator decorator, allowing the decorator to be used as: @decorator(with, arguments, and=kwargs) or @decorator ''' @wraps(f) def new_dec(*args, **kwargs):  if len(args) == 1 and len(kwargs) == 0 and callable(args[0]):   # actual decorated function   return f(args[0])  else:   # decorator arguments   return lambda realf: f(realf, *args, **kwargs)  return new_dec

使用起來很簡單,只要給裝飾器用@doublewrap裝飾一下,這個裝飾器就支持寫括號和不寫括號兩種寫法了。

def test_doublewrap(): from util import doublewrap from functools import wraps   @doublewrap def mult(f, factor=2):  '''multiply a function's return value'''  @wraps(f)  def wrap(*args, **kwargs):   return factor*f(*args,**kwargs)  return wrap  # try normal @mult def f(x, y):  return x + y  # try args @mult(3) def f2(x, y):  return x*y  # try kwargs @mult(factor=5) def f3(x, y):  return x - y  assert f(2,3) == 10 assert f2(2,5) == 30 assert f3(8,1) == 5*7

原理也不難,只有短短不到10行代碼。

裝飾器我們都知道,是用來處理一個函數(shù),返回一個新的函數(shù)的(如果你不理解裝飾器,可以看一下這個經(jīng)典的解釋)。

new_func = decorator(func)

我們使用的,就是被裝飾器裝飾的新函數(shù)了。裝飾器只是一個語法糖,其實它也是一個函數(shù),給它傳入一個函數(shù)作為參數(shù),就返回一個新的函數(shù)。那么既然裝飾器也是一個函數(shù),我們就可以用裝飾器裝飾這個函數(shù)。也就是,“裝飾器的裝飾器”。

裝飾器第一個參數(shù)肯定是原函數(shù),如果裝飾器可以接收參數(shù)的話,要么第一個參數(shù)是原函數(shù),后面跟別的參數(shù);要么就只有原函數(shù)一個參數(shù)。所以,我們這個“裝飾器的裝飾器”做的事情就是,判斷裝飾器接收的參數(shù),如果只有一個并且第一個參數(shù)是可調(diào)用的(callable),那么這就是一個無參數(shù)的裝飾器(不需要加括號)。如果還有別的參數(shù),就返回一個生成裝飾器的函數(shù)(decorator_maker)。

發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 台前县| 邢台县| 绩溪县| 阳谷县| 七台河市| 朝阳县| 木里| 安国市| 百色市| 屯昌县| 比如县| 右玉县| 绥滨县| 富锦市| 乌兰浩特市| 玉屏| 新丰县| 林口县| 阳朔县| 平度市| 肥城市| 嫩江县| 楚雄市| 蒙阴县| 会宁县| 云霄县| 阳朔县| 鲁山县| 天柱县| 昭通市| 丽江市| 荥经县| 资源县| 溧阳市| 霸州市| 育儿| 黔西县| 广灵县| 沭阳县| 连城县| 漯河市|