成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Python裝飾器-閉包與函數裝飾器

開發 前端
在不改變原函數代碼的情況下,給原函數增加了一些額外的功能,并且能夠通用于其他函數,這樣的函數就稱作為裝飾器。

一、閉包

在學習裝飾器前,需要先了解閉包的概念。形成閉包的要點:

  • 函數嵌套
  • 將內部函數作為外部函數的返回值
  • 內部函數必須要使用到外部函數的變量

下面以一個計算列表平均值的案例來講解閉包:

def make_average():
# 創建一個列表,用來保存數值
nums = []

# 定義一個內部函數,用來計算列表的平均值
def average(n):
# 將數值添加到列表中
nums.append(n)
# 返回平均值
return sum(nums) / len(nums)

return average
  1. 首先,定義一個函數make_average;
  2. 其次,在make_average函數內定義一個空列表,用來存儲數值;
  3. 再次,定義一個內部函數,用來計算列表平均值;
  4. 最后,將這個內函數作為外函數make_average的返回值,注意不要加( ),加( )就變成調用這個函數了。
# 調用外部函數,并將其復制給一個變量,注意:此時返回的是內函數的內存地址
a = make_average()
# 給這個變量加(),就相當于調用了內函數average
print(a(20))
print(a(30))

運行結果如下:當傳入的數值為20時,列表中只有一個數,所以計算結果是20;當再傳入一個數值30時,此時列表中就有兩個數20和30,所以平均值的計算結果是25.

二、裝飾器

1.裝飾器引入

例如,有以下兩個函數,分別計算兩個數的和以及成績:

def add(a, b):
"""計算兩數之和"""
res = a + b
return res

def mul(a, b):
"""計算兩數之積"""
res = a * b
return res

現在有個需求:我想要在每個函數的計算開始前打印“開始計算...”,在計算結束后打印“計算結束...”。我們可以通過直接修改函數代碼的方式來滿足這個需求,但這樣會面臨以下問題:

  1. 如果要修改的函數過多,十個甚至一百個函數,未免不現實;
  2. 不便于后期維護,例如我不想打印“開始計算...”了,而是要打印“begin...”,豈不是又要重新修改一遍;
  3. 違反開閉原則(OCP),即程序的設計,要求開放對程序的擴展、關閉對程序的修改;

所以,上述直接修改函數代碼的方式不可行。我們希望在不修改原函數的情況下,實現對函數的擴展。例如:

def new_add(a, b):
print("開始計算...")
r = add(a, b)
print("計算結束...")
return r


print(new_add(22, 33))

執行結果如下:

這種新創建一個函數的方式雖然沒有修改原函數,但面臨一個很嚴重的問題:

只能擴展指定函數,不能通用于其它函數,例如擴展上述的add函數,而不能擴展mul函數,如果想要擴展mul函數,只能再創建一個擴展函數;

因為,我們希望可以定義一個通用的擴展函數,可以作用域所有函數。這類不改變原函數代碼的通用函數就是:裝飾器。

2.函數裝飾器

裝飾器本質上是一個python函數或類,它可以讓其他函數或類在不需要做任何代碼修改的前提下增加額外功能,也就是為已經存在的對象添加額外功能,裝飾器的返回值也是一個函數/類對象。它經常用于有切面需求的場景,比如:插入日志、性能測試、事務處理、緩存、權限校驗等場景。

1)被裝飾函數不帶參數

例如:

def wrapper_info(func):
def inner():
print("開始介紹...")
res = func()
print("介紹結束...")
return res

return inner

def introduce1():
print("我是周潤發,我來自HONG KONG")

info = wrapper_info(introduce1)
info()

運行結果如下:

可見,在沒有改變原函數代碼的情況下,即給原函數增加了一些額外的功能,func是要修飾的函數,作為一個變量傳入裝飾函數,能夠通用于其他函數,這個wrapper_info就是裝飾器。但目前面臨的問題是,被裝飾函數如果帶參數怎么辦?例如:

def introduce2(name, age):
print(f"我叫{name}, 我今年{age}歲了")

2)被裝飾函數帶參數

盡管可以在裝飾器wrapper_info中傳入name、age,但并不是每個被裝飾的函數都只有name、age,亦或是指定類型的參數,還有可能傳入的是字典、列表、元組等。也就是傳入參數的類型和數量不固定怎么辦?

此時就需要用到不定長參數:(*args, **kwargs)

def wrapper_info(func):
"""
用來對其他函數進行擴展,使其他函數可以在執行前做一些額外的動作
:param func: 要擴展的函數對象
:return:
"""
def inner(*args, **kwargs):
print("開始介紹...")
res = func(*args, **kwargs)
print("介紹結束...")
return res

return inner

例如:

def introduce3(name, age, city):
print(f"我叫{name}, 我今年{age}歲了, 我來自{city}")

運行結果如下:

3)裝飾器帶參數

上述提到的是裝飾器,一種是應用于被裝飾的函數不帶參數,一種是被裝飾的函數帶參數,那裝飾器本身能否帶參數呢?比如我定義一個變量,想通過傳入不同的值來控制這個裝飾器實現不同的功能。答案是肯定的,例如:

def use_log(level):
def decorator(func):
def inner(*args, **kwargs):
if level == "warn":
logging.warning("%s is running by warning" % func.__name__)
elif level == "info":
logging.warning("%s is running by info" % func.__name__)
else:
logging.warning("%s is running by other" % func.__name__)
return func(*args, **kwargs)

return inner

return decorator


def introduce4(name, age, city):
print(f"我叫{name}, 我今年{age}歲了, 我來自{city}")


info1 = use_log(introduce4('周星馳', 28, '香港'))
info1('info')
info2 = use_log(introduce4('周潤發', 28, '香港'))
info2('warn')
info3 = use_log(introduce4('成龍', 28, '香港'))
info3('xxx')

運行結果如下:

3.裝飾器調用

方式一:以函數方式調用

info3 = wrapper_info(introduce3)
info3('劉德華', 28, '香港')

如果是裝飾器函數帶參數,則調用方式為:

info4 = use_log(introduce4('周星馳', 28, '香港'))
info4('info')

方式二:以語法糖方式調用

即在被裝飾函數上方以@符號進行修飾

@wrapper_info
def introduce3(name, age, city):
print(f"我叫{name}, 我今年{age}歲了, 我來自{city}")

introduce3('劉德華', 28, '香港')

如果是裝飾器函數帶參數,例如上述的use_log,則需要在裝飾器中傳入參數:

@use_log('info')
def introduce4(name, age, city):
print(f"我叫{name}, 我今年{age}歲了, 我來自{city}")

小結

什么是裝飾器?

在不改變原函數代碼的情況下,給原函數增加了一些額外的功能,并且能夠通用于其他函數,這樣的函數就稱作為裝飾器。

裝飾器的調用

可以通過傳統調用函數的方式進行調用,也可以通過@裝飾器的方式調用

裝飾器的特點

  • 通過裝飾器,可以在不修改原來函數的情況下對函數進行擴展
  • 一個函數可以同時指定多個裝飾器
責任編輯:武曉燕 來源: 今日頭條
相關推薦

2010-02-01 17:50:32

Python裝飾器

2021-06-01 07:19:58

Python函數裝飾器

2022-09-19 23:04:08

Python裝飾器語言

2016-11-01 09:24:38

Python裝飾器

2023-12-11 15:51:00

Python裝飾器代碼

2024-05-24 11:36:28

Python裝飾器

2024-02-26 00:00:00

TypeScript裝飾器decorators

2024-09-12 15:32:35

裝飾器Python

2024-09-03 09:47:09

閉包裝飾器Python

2022-10-21 07:50:35

裝飾器Python編程

2024-09-23 09:00:00

裝飾器函數代碼

2024-11-04 15:30:43

Python裝飾器函數

2025-01-22 15:58:46

2023-12-13 13:28:16

裝飾器模式Python設計模式

2021-04-11 08:21:20

Python@property裝飾器

2022-09-21 09:04:07

Python裝飾器

2023-08-07 16:07:42

2021-05-27 07:12:19

Python函數裝飾器

2022-09-27 11:01:08

Python裝飾器

2022-09-26 09:02:54

TS 裝飾器TypeScript
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产视频一视频二 | 精品99久久久久久 | 精品免费国产一区二区三区四区介绍 | 日本一道本 | 国产视频久久 | 天天曰天天干 | 亚洲国产精品一区二区久久 | 亚洲精品久久久久中文字幕欢迎你 | 国产精品不卡一区 | 在线观看午夜视频 | 亚洲一区精品视频 | 一区中文字幕 | 亚洲综合无码一区二区 | 黄网免费看| 久久久亚洲一区 | 精品久久电影 | 一级欧美一级日韩片 | 亚洲一区在线日韩在线深爱 | 99精品国产一区二区青青牛奶 | 成人在线播放网址 | 中国黄色毛片视频 | 成人不卡 | 欧美综合久久 | 麻豆91精品91久久久 | 国产成人精品免高潮在线观看 | 亚洲最大的成人网 | 成人免费看片网 | 一级欧美视频 | 91影视| av毛片 | 午夜精品一区二区三区在线 | 精品免费国产一区二区三区四区 | 免费在线观看av网址 | 久久久一区二区三区 | 精品九九九 | 精品视频在线观看 | 91精品在线播放 | 亚洲成人激情在线观看 | 精精国产xxxx视频在线播放7 | 色婷婷亚洲国产女人的天堂 | 神马影院一区二区三区 |