欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Python閉包與裝飾器原理及實(shí)例解析

 更新時(shí)間:2020年04月30日 11:03:51   作者:杰鑫哥  
這篇文章主要介紹了Python閉包與裝飾器原理及實(shí)例解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下

一、閉包

閉包相當(dāng)于函數(shù)中,嵌套另一個(gè)函數(shù),并返回。代碼如下:

def func(name): # 定義外層函數(shù)
  def inner_func(age): # 內(nèi)層函數(shù)
    print('name: ', name, ', age: ', age)
  return inner_func # 注意此處要返回,才能體現(xiàn)閉包

bb = func('jayson') # 將字符串傳給func函數(shù),返回inner_func并賦值給變量
bb(28) # 通過(guò)變量調(diào)用func函數(shù),傳入?yún)?shù),從而完成閉包
>>
name: jayson , age: 28

二、裝飾器

裝飾器:把函數(shù)test當(dāng)成變量傳入裝飾函數(shù)deco --> 執(zhí)行了裝飾操作后,變量傳回給了函數(shù)test()。比如裝飾器效果是test = test-1,test函數(shù)經(jīng)過(guò)deco裝飾后,調(diào)用test其實(shí)執(zhí)行的是 test = test-1。

1、裝飾器是利用閉包原理,區(qū)別是裝飾器在閉包中傳入的參數(shù)是函數(shù),而不是變量。

注:其實(shí)在裝飾器中,函數(shù)即變量

def deco(func): # 傳入func函數(shù)。
  print('decoration')
  return func
def test():
  print('test_func')

test = deco(test) # 對(duì)函數(shù)進(jìn)行裝飾。執(zhí)行了deco函數(shù),并將返回值賦值給test
>>
# 輸出deco的運(yùn)行結(jié)果
decoration

test() # 運(yùn)行裝飾后的函數(shù)
>>
test_func

2、以上代碼等價(jià)于

def deco(func): # 傳入func函數(shù)。
  print('decoration')
  return func

@deco # 等價(jià)于上一代碼中test = deco(test),不過(guò)上一代碼需放在定義test之后
def test():
  print('test_func')

>>
# 輸出deco的運(yùn)行結(jié)果
decoration

test() # 運(yùn)行裝飾后的函數(shù)
>>
test_func

3、裝飾器(簡(jiǎn)版)

def deco(func): # 裝飾函數(shù)傳入func
  print('decoration')
  return func

@deco # 裝飾函數(shù)。
def test():
  print('test_func') 
# 定義完函數(shù)后,會(huì)直接執(zhí)行裝飾器deco(test)
>>
decoration

# 調(diào)用test,執(zhí)行test函數(shù)
test()
>> 
test_func

3、裝飾器(升級(jí)版)

在上一個(gè)版本中,由于在定義裝飾器 + 函數(shù)時(shí),就會(huì)執(zhí)行裝飾函數(shù)里面的語(yǔ)句。

為了使其在未被調(diào)用時(shí)候不執(zhí)行,需要再嵌套一個(gè)函數(shù),將函數(shù)進(jìn)行包裹。

def deco(func): 
  print('decoration') # 此處未調(diào)用func函數(shù)時(shí),會(huì)直接執(zhí)行
  def wrapper(): # 名稱自定義,一般用wrapper
    print('execute') # 此處未調(diào)用func函數(shù)時(shí),不會(huì)執(zhí)行
    func() # 執(zhí)行函數(shù)
  return wrapper # 此處返回wrapper給func,通過(guò)外部func()執(zhí)行

@deco # 注意:此處不能有括號(hào)。有括號(hào)的形式是func未傳入最外層deco(),傳入deco的子函數(shù)中
def test():
  print('test_func')
>>
decoration
#調(diào)用test
test()
>>
execute
test_func

注意:如果func函數(shù)本身有返回值,同樣需要在包裹函數(shù)中返回

def deco(func): 
  print('decoration')
  def wrapper():
    print('execute')
    a = func() # 執(zhí)行函數(shù),并返回值
    print('done')
    return a # 將func的返回值一并返回
  return wrapper

@deco
def test():
  print('test_func')
  return 5 # 增加返回值
>>
decoration

#調(diào)用test
test()
>>
execute
test_func
done
 # 此處是test函數(shù)的返回值

3、裝飾器(進(jìn)階版)

在包裹函數(shù)中,參數(shù)形式設(shè)置為*arg、**kwarg,會(huì)使得函數(shù)更加靈活。

當(dāng)修改test函數(shù)參數(shù)形式時(shí),不用在裝飾器中同時(shí)修改。

import time

def deco(func):
  def inner(*arg, **kwarg): # 此處傳入?yún)?shù)
    begin_time = time.time()
    time.sleep(2)
    a = func(*arg, **kwarg) # 調(diào)用函數(shù),使用傳入的參數(shù)
    end_time = time.time()
    print('運(yùn)行時(shí)間:', end_time - begin_time)
    return a
  return inner

@deco
def test(a):
  print('test function:', a)
  return a

# 調(diào)用函數(shù)
test(5)
>>
test function: 5
運(yùn)行時(shí)間: 2.0003252029418945
 # 5是函數(shù)返回的值

4、高階版

有時(shí)候我們會(huì)發(fā)現(xiàn)有的裝飾器帶括號(hào),其原因是將上述的裝飾器外面又套了一個(gè)函數(shù)

import time

def outer(): # 在原裝飾器外套一層函數(shù),將裝飾器封裝在函數(shù)里面。(outer自定義)
  def deco(func): # 原裝飾器,后面的代碼一樣
    def inner(*arg, **kwarg): 
      begin_time = time.time()
      time.sleep(2)
      a = func(*arg, **kwarg) 
      end_time = time.time()
      print('運(yùn)行時(shí)間:', end_time - begin_time)
      return a
    return inner
  return deco # 注意:此處需返回裝飾函數(shù)

@outer() # 此處就需要加括號(hào),其實(shí)是調(diào)用了outer()函數(shù),將test傳進(jìn)其子函數(shù)
def test(a):
  print('test function:', a)
  return a

test(4)
>>
test function: 4
運(yùn)行時(shí)間: 2.000566005706787
 # 返回4

5、高階終結(jié)版

帶參數(shù)的裝飾器(裝飾器加括號(hào),帶參數(shù))

import time

def outer(choose): # 在最外層函數(shù)中加入?yún)?shù)
  if choose==1: # 通過(guò)choose參數(shù),選擇裝飾器
    def deco(func):
      def inner(*arg, **kwarg):
        print('decoration1')
        begin_time = time.time()
        time.sleep(2) # 睡眠2s
        a = func(*arg, **kwarg) 
        end_time = time.time()
        print('運(yùn)行時(shí)間1:', end_time - begin_time)
        return a
      return inner
    return deco
  
  else:
    def deco(func):
      def inner(*arg, **kwarg): 
        print('decoration2')
        begin_time = time.time()
        time.sleep(5) # 睡眠5s
        a = func(*arg, **kwarg) 
        end_time = time.time()
        print('運(yùn)行時(shí)間2:', end_time - begin_time)
        return a
      return inner
    return deco

@outer(1) # 由于outer中有參數(shù),此處必須傳入?yún)?shù)
def test1(a):
  print('test function1:', a)
  return a

@outer(5) # 傳入另一個(gè)參數(shù)
def test2(a):
  print('test function2:', a)
  return a


# 分別調(diào)用2個(gè)函數(shù)(2個(gè)函數(shù)裝飾器相同,裝飾器參數(shù)不同)
test1(2) # 調(diào)用test1
>>
decoration1
test function1: 2
運(yùn)行時(shí)間1: 2.000072717666626 # 2秒
 # test1的返回值

test2(4) # 調(diào)用test2
>>
decoration2
test function2: 4
運(yùn)行時(shí)間2: 5.000797986984253 # 5秒
 # test2的返回值

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Pandas 合并多個(gè)Dataframe(merge,concat)的方法

    Pandas 合并多個(gè)Dataframe(merge,concat)的方法

    今天小編就為大家分享一篇Pandas 合并多個(gè)Dataframe(merge,concat)的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2018-06-06
  • Python 如何查找特定類型文件

    Python 如何查找特定類型文件

    這篇文章主要介紹了Python 如何定位特定類型文件,幫助大家更好的理解和學(xué)習(xí)python,感興趣的朋友可以了解下
    2020-08-08
  • 聊聊Numpy.array中[:]和[::]的區(qū)別在哪

    聊聊Numpy.array中[:]和[::]的區(qū)別在哪

    這篇文章主要介紹了在Numpy.array中[:]和[::]的區(qū)別說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-05-05
  • python中?@的含義以及基本使用方法

    python中?@的含義以及基本使用方法

    @用做函數(shù)的修飾符,可以在模塊或者類的定義層內(nèi)對(duì)函數(shù)進(jìn)行修飾,下面這篇文章主要給大家介紹了關(guān)于python中?@?的含義以及基本使用的相關(guān)資料,需要的朋友可以參考下
    2021-12-12
  • Python實(shí)現(xiàn)將多張圖片合成視頻并加入背景音樂(lè)

    Python實(shí)現(xiàn)將多張圖片合成視頻并加入背景音樂(lè)

    這篇文章主要為大家介紹了如何利用Python實(shí)現(xiàn)將多張圖片合成mp4視頻,并加入背景音樂(lè)。文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下
    2022-04-04
  • Python時(shí)間序列數(shù)據(jù)的預(yù)處理方法總結(jié)

    Python時(shí)間序列數(shù)據(jù)的預(yù)處理方法總結(jié)

    這篇文章主要介紹了Python時(shí)間序列數(shù)據(jù)的預(yù)處理方法總結(jié),時(shí)間序列數(shù)據(jù)隨處可見(jiàn),要進(jìn)行時(shí)間序列分析,我們必須先對(duì)數(shù)據(jù)進(jìn)行預(yù)處理。時(shí)間序列預(yù)處理技術(shù)對(duì)數(shù)據(jù)建模的準(zhǔn)確性有重大影響
    2022-07-07
  • PyQt使用QPropertyAnimation開(kāi)發(fā)簡(jiǎn)單動(dòng)畫(huà)

    PyQt使用QPropertyAnimation開(kāi)發(fā)簡(jiǎn)單動(dòng)畫(huà)

    這篇文章主要介紹了PyQt使用QPropertyAnimation開(kāi)發(fā)簡(jiǎn)單動(dòng)畫(huà),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-04-04
  • python逐行讀取文件內(nèi)容的三種方法

    python逐行讀取文件內(nèi)容的三種方法

    這篇文章主要介紹了python逐行讀取文件內(nèi)容的三種方法,非常的簡(jiǎn)單,下面直接看代碼吧
    2014-01-01
  • python uuid生成唯一id或str的最簡(jiǎn)單案例

    python uuid生成唯一id或str的最簡(jiǎn)單案例

    這篇文章主要介紹了python uuid生成唯一id或str的最簡(jiǎn)單案例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2021-01-01
  • python3?cookbook解壓可迭代對(duì)象賦值給多個(gè)變量的問(wèn)題及解決方案

    python3?cookbook解壓可迭代對(duì)象賦值給多個(gè)變量的問(wèn)題及解決方案

    這篇文章主要介紹了python3?cookbook-解壓可迭代對(duì)象賦值給多個(gè)變量,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2024-01-01

最新評(píng)論