溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

怎么在python中使用裝飾器

發(fā)布時(shí)間:2021-03-31 17:05:19 來源:億速云 閱讀:126 作者:Leah 欄目:開發(fā)技術(shù)

今天就跟大家聊聊有關(guān)怎么在python中使用裝飾器,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。

1、必備

#### 第一波 ####
def foo():
  print 'foo'
 
foo   #表示是函數(shù)
foo()  #表示執(zhí)行foo函數(shù)
 
#### 第二波 ####
def foo():
  print 'foo'
 
foo = lambda x: x + 1
foo()  # 執(zhí)行下面的lambda表達(dá)式,而不再是原來的foo函數(shù),因?yàn)楹瘮?shù) foo 被重新定義了

2、需求來了

初創(chuàng)公司有N個(gè)業(yè)務(wù)部門,1個(gè)基礎(chǔ)平臺(tái)部門,基礎(chǔ)平臺(tái)負(fù)責(zé)提供底層的功能,如:數(shù)據(jù)庫(kù)操作、redis調(diào)用、監(jiān)控API等功能。業(yè)務(wù)部門使用基礎(chǔ)功能時(shí),只需調(diào)用基礎(chǔ)平臺(tái)提供的功能即可。如下:

############### 基礎(chǔ)平臺(tái)提供的功能如下 ###############
def f1():
  print 'f1'
 
def f2():
  print 'f2'
 
def f3():
  print 'f3'
 
def f4():
  print 'f4'
 
############### 業(yè)務(wù)部門A 調(diào)用基礎(chǔ)平臺(tái)提供的功能 ###############
f1()
f2()
f3()
f4()
 
############### 業(yè)務(wù)部門B 調(diào)用基礎(chǔ)平臺(tái)提供的功能 ###############
f1()
f2()
f3()
f4()

目前公司有條不紊的進(jìn)行著,但是,以前基礎(chǔ)平臺(tái)的開發(fā)人員在寫代碼時(shí)候沒有關(guān)注驗(yàn)證相關(guān)的問題,即:基礎(chǔ)平臺(tái)的提供的功能可以被任何人使用?,F(xiàn)在需要對(duì)基礎(chǔ)平臺(tái)的所有功能進(jìn)行重構(gòu),為平臺(tái)提供的所有功能添加驗(yàn)證機(jī)制,即:執(zhí)行功能前,先進(jìn)行驗(yàn)證。

老大把工作交給 Low B,他是這么做的:

跟每個(gè)業(yè)務(wù)部門交涉,每個(gè)業(yè)務(wù)部門自己寫代碼,調(diào)用基礎(chǔ)平臺(tái)的功能之前先驗(yàn)證。誒,這樣一來基礎(chǔ)平臺(tái)就不需要做任何修改了。

當(dāng)天Low B 被開除了...

老大把工作交給 Low BB,他是這么做的:

只對(duì)基礎(chǔ)平臺(tái)的代碼進(jìn)行重構(gòu),讓N業(yè)務(wù)部門無需做任何修改

############### 基礎(chǔ)平臺(tái)提供的功能如下 ############### 

def f1():
  # 驗(yàn)證1
  # 驗(yàn)證2
  # 驗(yàn)證3
  print 'f1'

def f2():
  # 驗(yàn)證1
  # 驗(yàn)證2
  # 驗(yàn)證3
  print 'f2'

def f3():
  # 驗(yàn)證1
  # 驗(yàn)證2
  # 驗(yàn)證3
  print 'f3'

def f4():
  # 驗(yàn)證1
  # 驗(yàn)證2
  # 驗(yàn)證3
  print 'f4'
############### 業(yè)務(wù)部門不變 ############### 
### 業(yè)務(wù)部門A 調(diào)用基礎(chǔ)平臺(tái)提供的功能### 

f1()
f2()
f3()
f4()

### 業(yè)務(wù)部門B 調(diào)用基礎(chǔ)平臺(tái)提供的功能 ### 

f1()
f2()
f3()
f4()

過了一周 Low BB 被開除了...

老大把工作交給 Low BBB,他是這么做的:

只對(duì)基礎(chǔ)平臺(tái)的代碼進(jìn)行重構(gòu),其他業(yè)務(wù)部門無需做任何修改

############### 基礎(chǔ)平臺(tái)提供的功能如下 ############### 

def check_login():
  # 驗(yàn)證1
  # 驗(yàn)證2
  # 驗(yàn)證3
  pass

def f1():
  
  check_login()

  print 'f1'

def f2():
  
  check_login()

  print 'f2'

def f3():
  
  check_login()

  print 'f3'

def f4():
  
  check_login()
  
  print 'f4'

老大看了下Low BBB 的實(shí)現(xiàn),嘴角漏出了一絲的欣慰的笑,語(yǔ)重心長(zhǎng)的跟Low BBB聊了個(gè)天:

老大說:

寫代碼要遵循開發(fā)封閉原則,雖然在這個(gè)原則是用的面向?qū)ο箝_發(fā),但是也適用于函數(shù)式編程,簡(jiǎn)單來說,它規(guī)定已經(jīng)實(shí)現(xiàn)的功能代碼不允許被修改,但可以被擴(kuò)展,即:

封閉:已實(shí)現(xiàn)的功能代碼塊
開放:對(duì)擴(kuò)展開發(fā)

如果將開放封閉原則應(yīng)用在上述需求中,那么就不允許在函數(shù) f1 、f2、f3、f4的內(nèi)部進(jìn)行修改代碼,老板就給了Low BBB一個(gè)實(shí)現(xiàn)方案:

def w1(func):
  def inner():
    # 驗(yàn)證1
    # 驗(yàn)證2
    # 驗(yàn)證3
    return func()
  return inner
 
@w1
def f1():
  print 'f1'
@w1
def f2():
  print 'f2'
@w1
def f3():
  print 'f3'
@w1
def f4():
  print 'f4'

對(duì)于上述代碼,也是僅僅對(duì)基礎(chǔ)平臺(tái)的代碼進(jìn)行修改,就可以實(shí)現(xiàn)在其他人調(diào)用函數(shù) f1 f2 f3 f4 之前都進(jìn)行【驗(yàn)證】操作,并且其他業(yè)務(wù)部門無需做任何操作。

Low BBB心驚膽戰(zhàn)的問了下,這段代碼的內(nèi)部執(zhí)行原理是什么呢?

老大正要生氣,突然Low BBB的手機(jī)掉到地上,恰恰屏保就是Low BBB的女友照片,老大一看一緊一抖,喜笑顏開,交定了Low BBB這個(gè)朋友。詳細(xì)的開始講解了:

單獨(dú)以f1為例:

def w1(func):
  def inner():
    # 驗(yàn)證1
    # 驗(yàn)證2
    # 驗(yàn)證3
    return func()
  return inner
@w1
def f1():
  print 'f1'

當(dāng)寫完這段代碼后(函數(shù)未被執(zhí)行、未被執(zhí)行、未被執(zhí)行),python解釋器就會(huì)從上到下解釋代碼,步驟如下:

def w1(func): ==>將w1函數(shù)加載到內(nèi)存
@w1

沒錯(cuò),從表面上看解釋器僅僅會(huì)解釋這兩句代碼,因?yàn)楹瘮?shù)在沒有被調(diào)用之前其內(nèi)部代碼不會(huì)被執(zhí)行。

從表面上看解釋器著實(shí)會(huì)執(zhí)行這兩句,但是 @w1 這一句代碼里卻有大文章,@函數(shù)名 是python的一種語(yǔ)法糖。

如上例@w1內(nèi)部會(huì)執(zhí)行一下操作:

  • 執(zhí)行w1函數(shù),并將 @w1 下面的 函數(shù) 作為w1函數(shù)的參數(shù),即:@w1 等價(jià)于 w1(f1)
     所以,內(nèi)部就會(huì)去執(zhí)行:

      def inner:
        #驗(yàn)證
        return f1()  # func是參數(shù),此時(shí) func 等于 f1
      return inner   # 返回的 inner,inner代表的是函數(shù),非執(zhí)行函數(shù)

     

  • 其實(shí)就是將原來的 f1 函數(shù)塞進(jìn)另外一個(gè)函數(shù)中

  • 將執(zhí)行完的 w1 函數(shù)返回值賦值給@w1下面的函數(shù)的函數(shù)名
     
    w1函數(shù)的返回值是:
        def inner:
             #驗(yàn)證
             return 原來f1()  # 此處的 f1 表示原來的f1函數(shù)
     然后,將此返回值再重新賦值給 f1,即:
     
    新f1 = def inner:
                 #驗(yàn)證
                 return 原來f1() 
     所以,以后業(yè)務(wù)部門想要執(zhí)行 f1 函數(shù)時(shí),就會(huì)執(zhí)行 新f1 函數(shù),在 新f1 函數(shù)內(nèi)部先執(zhí)行驗(yàn)證,再執(zhí)行原來的f1函數(shù),然后將 原來f1 函數(shù)的返回值 返回給了業(yè)務(wù)調(diào)用者。
     如此一來, 即執(zhí)行了驗(yàn)證的功能,又執(zhí)行了原來f1函數(shù)的內(nèi)容,并將原f1函數(shù)返回值 返回給業(yè)務(wù)調(diào)用著

Low BBB 你明白了嗎?要是沒明白的話,我晚上去你家?guī)湍憬鉀Q吧!??!

先把上述流程看懂,之后還會(huì)繼續(xù)更新...

3、問答時(shí)間

問題:被裝飾的函數(shù)如果有參數(shù)呢?

一個(gè)參數(shù):

def w1(func):
  def inner(arg):
    # 驗(yàn)證1
    # 驗(yàn)證2
    # 驗(yàn)證3
    return func(arg)
  return inner

@w1
def f1(arg):
  print 'f1'

兩個(gè)參數(shù):

def w1(func):
  def inner(arg1,arg2):
    # 驗(yàn)證1
    # 驗(yàn)證2
    # 驗(yàn)證3
    return func(arg1,arg2)
  return inner

@w1
def f1(arg1,arg2):
  print 'f1'

三個(gè)參數(shù):

def w1(func):
  def inner(arg1,arg2,arg3):
    # 驗(yàn)證1
    # 驗(yàn)證2
    # 驗(yàn)證3
    return func(arg1,arg2,arg3)
  return inner

@w1
def f1(arg1,arg2,arg3):
  print 'f1'

問題:可以裝飾具有處理n個(gè)參數(shù)的函數(shù)的裝飾器?

def w1(func):
  def inner(*args,**kwargs):
    # 驗(yàn)證1
    # 驗(yàn)證2
    # 驗(yàn)證3
    return func(*args,**kwargs)
  return inner
 
@w1
def f1(arg1,arg2,arg3):
  print 'f1'

問題:一個(gè)函數(shù)可以被多個(gè)裝飾器裝飾嗎?

def w1(func):
  def inner(*args,**kwargs):
    # 驗(yàn)證1
    # 驗(yàn)證2
    # 驗(yàn)證3
    return func(*args,**kwargs)
  return inner
 
def w2(func):
  def inner(*args,**kwargs):
    # 驗(yàn)證1
    # 驗(yàn)證2
    # 驗(yàn)證3
    return func(*args,**kwargs)
  return inner
 
@w1
@w2
def f1(arg1,arg2,arg3):
  print 'f1'

問題:還有什么更吊的裝飾器嗎?

#!/usr/bin/env python
#coding:utf-8
 
def Before(request,kargs):
  print 'before'
   
def After(request,kargs):
  print 'after'
 
 
def Filter(before_func,after_func):
  def outer(main_func):
    def wrapper(request,kargs):
       
      before_result = before_func(request,kargs)
      if(before_result != None):
        return before_result;
       
      main_result = main_func(request,kargs)
      if(main_result != None):
        return main_result;
       
      after_result = after_func(request,kargs)
      if(after_result != None):
        return after_result;
       
    return wrapper
  return outer
   
@Filter(Before, After)
def Index(request,kargs):
  print 'index'

4、functools.wraps

上述的裝飾器雖然已經(jīng)完成了其應(yīng)有的

功能,即:裝飾器內(nèi)的函數(shù)代指了原函數(shù),注意其只是代指而非相等,原函數(shù)的元信息沒有被賦值到裝飾器函數(shù)內(nèi)部。例如:函數(shù)的注釋信息

無元信息:

def outer(func):
  def inner(*args, **kwargs):
    print(inner.__doc__) # None
    return func()
  return inner

@outer
def function():
  """
  asdfasd
  :return:
  """
  print('func')

如果使用@functools.wraps裝飾裝飾器內(nèi)的函數(shù),那么就會(huì)代指元信息和函數(shù)。

含元信息:

def outer(func):
  @functools.wraps(func)
  def inner(*args, **kwargs):
    print(inner.__doc__) # None
    return func()
  return inner

@outer
def function():
  """
  asdfasd
  :return:
  """
  print('func')

看完上述內(nèi)容,你們對(duì)怎么在python中使用裝飾器有進(jìn)一步的了解嗎?如果還想了解更多知識(shí)或者相關(guān)內(nèi)容,請(qǐng)關(guān)注億速云行業(yè)資訊頻道,感謝大家的支持。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI