千鋒教育-做有情懷、有良心、有品質(zhì)的職業(yè)教育機(jī)構(gòu)

手機(jī)站
千鋒教育

千鋒學(xué)習(xí)站 | 隨時隨地免費(fèi)學(xué)

千鋒教育

掃一掃進(jìn)入千鋒手機(jī)站

領(lǐng)取全套視頻
千鋒教育

關(guān)注千鋒學(xué)習(xí)站小程序
隨時隨地免費(fèi)學(xué)習(xí)課程

當(dāng)前位置:首頁  >  技術(shù)干貨  > python自定義裝飾器

python自定義裝飾器

來源:千鋒教育
發(fā)布人:xqq
時間: 2024-03-05 02:01:33 1709575293

**Python自定義裝飾器:優(yōu)雅而強(qiáng)大的函數(shù)修飾工具**

_x000D_

**引言**

_x000D_

Python自定義裝飾器是一種強(qiáng)大而優(yōu)雅的函數(shù)修飾工具,它能夠在不修改原函數(shù)代碼的情況下,為函數(shù)添加額外的功能。裝飾器可以用于日志記錄、性能分析、權(quán)限驗證等各種場景,極大地提高了代碼的可復(fù)用性和可維護(hù)性。本文將深入探討Python自定義裝飾器的原理、應(yīng)用和一些常見問題。

_x000D_

**Python自定義裝飾器的原理**

_x000D_

裝飾器本質(zhì)上是一個函數(shù),它接受一個函數(shù)作為參數(shù),并返回一個新的函數(shù)。通過裝飾器,我們可以在不修改原函數(shù)代碼的情況下,在函數(shù)的前后添加額外的邏輯。裝飾器使用@符號將修飾器函數(shù)應(yīng)用到目標(biāo)函數(shù)上,使得目標(biāo)函數(shù)的調(diào)用會經(jīng)過修飾器函數(shù)。

_x000D_

下面是一個簡單的裝飾器示例:

_x000D_

`python

_x000D_

def logger(func):

_x000D_

def wrapper(*args, **kwargs):

_x000D_

print(f"Calling function: {func.__name__}")

_x000D_

return func(*args, **kwargs)

_x000D_

return wrapper

_x000D_

@logger

_x000D_

def add(a, b):

_x000D_

return a + b

_x000D_

result = add(1, 2) # 輸出:Calling function: add

_x000D_

print(result) # 輸出:3

_x000D_ _x000D_

在上述示例中,logger是一個裝飾器函數(shù),它接受一個函數(shù)作為參數(shù),并返回一個新的函數(shù)wrapperwrapper函數(shù)在調(diào)用目標(biāo)函數(shù)add之前,先打印了一條日志,然后再調(diào)用目標(biāo)函數(shù),并返回其結(jié)果。通過在add函數(shù)上添加@logger裝飾器,我們實現(xiàn)了在函數(shù)調(diào)用前打印日志的功能。

_x000D_

**Python自定義裝飾器的應(yīng)用**

_x000D_

1. **日志記錄**

_x000D_

在實際開發(fā)中,我們經(jīng)常需要記錄函數(shù)的調(diào)用信息,以便于調(diào)試和追蹤問題。使用裝飾器可以簡化日志記錄的代碼,并將其應(yīng)用于多個函數(shù)。

_x000D_

`python

_x000D_

def logger(func):

_x000D_

def wrapper(*args, **kwargs):

_x000D_

print(f"Calling function: {func.__name__}")

_x000D_

return func(*args, **kwargs)

_x000D_

return wrapper

_x000D_

@logger

_x000D_

def add(a, b):

_x000D_

return a + b

_x000D_

@logger

_x000D_

def subtract(a, b):

_x000D_

return a - b

_x000D_

result1 = add(1, 2) # 輸出:Calling function: add

_x000D_

result2 = subtract(3, 2) # 輸出:Calling function: subtract

_x000D_ _x000D_

2. **性能分析**

_x000D_

裝飾器還可以用于性能分析,幫助我們找出代碼中的性能瓶頸。下面是一個簡單的性能分析裝飾器示例:

_x000D_

`python

_x000D_

import time

_x000D_

def timer(func):

_x000D_

def wrapper(*args, **kwargs):

_x000D_

start_time = time.time()

_x000D_

result = func(*args, **kwargs)

_x000D_

end_time = time.time()

_x000D_

print(f"Function {func.__name__} took {end_time - start_time:.2f} seconds")

_x000D_

return result

_x000D_

return wrapper

_x000D_

@timer

_x000D_

def fib(n):

_x000D_

if n <= 1:

_x000D_

return n

_x000D_

return fib(n-1) + fib(n-2)

_x000D_

result = fib(10) # 輸出:Function fib took 0.00 seconds

_x000D_ _x000D_

在上述示例中,timer裝飾器用于計算函數(shù)的執(zhí)行時間,并在函數(shù)調(diào)用完成后打印出來。通過在fib函數(shù)上添加@timer裝飾器,我們可以方便地獲取fib函數(shù)的執(zhí)行時間。

_x000D_

3. **權(quán)限驗證**

_x000D_

裝飾器還可以用于權(quán)限驗證,只有滿足特定條件的用戶才能調(diào)用被裝飾的函數(shù)。下面是一個簡單的權(quán)限驗證裝飾器示例:

_x000D_

`python

_x000D_

def login_required(func):

_x000D_

def wrapper(*args, **kwargs):

_x000D_

if is_logged_in():

_x000D_

return func(*args, **kwargs)

_x000D_

else:

_x000D_

raise Exception("Login required")

_x000D_

return wrapper

_x000D_

@login_required

_x000D_

def delete_file(file_path):

_x000D_

# 刪除文件的邏輯

_x000D_

delete_file("/path/to/file") # 如果未登錄,則拋出異常

_x000D_ _x000D_

在上述示例中,login_required裝飾器用于驗證用戶是否已登錄,只有已登錄的用戶才能調(diào)用delete_file函數(shù)。通過在delete_file函數(shù)上添加@login_required裝飾器,我們實現(xiàn)了權(quán)限驗證的功能。

_x000D_

**Python自定義裝飾器的常見問題**

_x000D_

1. **裝飾器是否可以帶參數(shù)?**

_x000D_

是的,裝飾器可以帶參數(shù)。如果裝飾器本身需要接受參數(shù),則需要編寫一個額外的函數(shù),該函數(shù)用于接受裝飾器參數(shù),并返回一個裝飾器函數(shù)。

_x000D_

`python

_x000D_

def logger(level):

_x000D_

def decorator(func):

_x000D_

def wrapper(*args, **kwargs):

_x000D_

print(f"[{level}] Calling function: {func.__name__}")

_x000D_

return func(*args, **kwargs)

_x000D_

return wrapper

_x000D_

return decorator

_x000D_

@logger(level="INFO")

_x000D_

def add(a, b):

_x000D_

return a + b

_x000D_

result = add(1, 2) # 輸出:[INFO] Calling function: add

_x000D_ _x000D_

在上述示例中,logger函數(shù)是一個裝飾器工廠函數(shù),它接受一個參數(shù)level,并返回一個裝飾器函數(shù)decoratordecorator函數(shù)用于接受目標(biāo)函數(shù),并返回一個新的函數(shù)wrapper。通過在add函數(shù)上添加@logger(level="INFO")裝飾器,我們實現(xiàn)了在函數(shù)調(diào)用前打印日志,并指定了日志級別為"INFO"。

_x000D_

2. **裝飾器是否會改變被修飾函數(shù)的元信息?**

_x000D_

裝飾器會改變被修飾函數(shù)的元信息。在使用裝飾器修飾函數(shù)時,函數(shù)的__name____doc__等屬性會發(fā)生變化,這可能會對一些依賴于這些屬性的代碼造成影響。為了解決這個問題,可以使用functools.wraps裝飾器來保留原函數(shù)的元信息。

_x000D_

`python

_x000D_

import functools

_x000D_

def logger(func):

_x000D_

@functools.wraps(func)

_x000D_

def wrapper(*args, **kwargs):

_x000D_

print(f"Calling function: {func.__name__}")

_x000D_

return func(*args, **kwargs)

_x000D_

return wrapper

_x000D_

@logger

_x000D_

def add(a, b):

_x000D_

"""Add two numbers"""

_x000D_

return a + b

_x000D_

print(add.__name__) # 輸出:add

_x000D_

print(add.__doc__) # 輸出:Add two numbers

_x000D_ _x000D_

在上述示例中,functools.wraps裝飾器用于將wrapper函數(shù)的元信息設(shè)置為原函數(shù)add的元信息。通過使用@functools.wraps(func)修飾wrapper函數(shù),我們保留了add函數(shù)的元信息,使其在被調(diào)用時與原函數(shù)一致。

_x000D_

**總結(jié)**

_x000D_

Python自定義裝飾器是一種強(qiáng)大而優(yōu)雅的函數(shù)修飾工具,能夠在不修改原函數(shù)代碼的情況下,為函數(shù)添加額外的功能。裝飾器可以應(yīng)用于日志記錄、性能分析、權(quán)限驗證等各種場景,提高了代碼的可復(fù)用性和可維護(hù)性。相信讀者對Python自定義裝飾器有了更深入的了解,并能夠靈活運(yùn)用于實際開發(fā)中。

_x000D_

**問答擴(kuò)展**

_x000D_

1. **裝飾器和函數(shù)裝飾器有什么區(qū)別?**

_x000D_

裝飾器是一種特殊的函數(shù),它接受一個函數(shù)作為參數(shù),并返回一個新的函數(shù)。裝飾器可以通過@符號將修飾器函數(shù)應(yīng)用到目標(biāo)函數(shù)上,使得目標(biāo)函數(shù)的調(diào)用會經(jīng)過修飾器函數(shù)。

_x000D_

函數(shù)裝飾器是一種特殊的裝飾器,它用于修飾函數(shù)。函數(shù)裝飾器可以在函數(shù)定義前使用,也可以在函數(shù)定義后使用。函數(shù)裝飾器的語法糖形式是@decorator,其中decorator是一個裝飾器函數(shù)。

_x000D_

2. **裝飾器可以嵌套使用嗎?**

_x000D_

是的,裝飾器可以嵌套使用。當(dāng)多個裝飾器應(yīng)用于同一個函數(shù)時,它們會按照從上到下的順序依次生效。

_x000D_

`python

_x000D_

def decorator1(func):

_x000D_

def wrapper(*args, **kwargs):

_x000D_

print("Decorator 1")

_x000D_

return func(*args, **kwargs)

_x000D_

return wrapper

_x000D_

def decorator2(func):

_x000D_

def wrapper(*args, **kwargs):

_x000D_

print("Decorator 2")

_x000D_

return func(*args, **kwargs)

_x000D_

return wrapper

_x000D_

@decorator1

_x000D_

@decorator2

_x000D_

def add(a, b):

_x000D_

return a + b

_x000D_

result = add(1, 2) # 輸出:Decorator 1

_x000D_

# 輸出:Decorator 2

_x000D_

print(result) # 輸出:3

_x000D_ _x000D_

在上述示例中,add函數(shù)先經(jīng)過decorator2裝飾器修飾,然后再經(jīng)過decorator1裝飾器修飾。調(diào)用add函數(shù)時,會先打印出"Decorator 1",然后再打印出"Decorator 2"。

_x000D_

3. **裝飾器是否可以取消或移除?**

_x000D_

裝飾器本質(zhì)上是一個函數(shù),因此可以通過重新定義原函數(shù)來取消或移除裝飾器的效果。在重新定義原函數(shù)時,需要將其恢復(fù)為未被裝飾的狀態(tài)。

_x000D_

`python

_x000D_

def logger(func):

_x000D_

def wrapper(*args, **kwargs):

_x000D_

print(f"Calling function: {func.__name__}")

_x000D_

return func(*args, **kwargs)

_x000D_

return wrapper

_x000D_

@logger

_x000D_

def add(a, b):

_x000D_

return a + b

_x000D_

# 取消裝飾器的效果

_x000D_

add = add.__wrapped__

_x000D_

result = add(1, 2) # 不會打印日志

_x000D_

print(result) # 輸出:3

_x000D_ _x000D_

在上述示例中,add函數(shù)經(jīng)過@logger裝飾器修飾,會在調(diào)用前打印日志。通過將add重新賦值為add.__wrapped__,我們將其恢復(fù)為未被裝飾的狀態(tài),從而取消了裝飾器的效果。

_x000D_
tags: python教程
聲明:本站稿件版權(quán)均屬千鋒教育所有,未經(jīng)許可不得擅自轉(zhuǎn)載。
10年以上業(yè)內(nèi)強(qiáng)師集結(jié),手把手帶你蛻變精英
請您保持通訊暢通,專屬學(xué)習(xí)老師24小時內(nèi)將與您1V1溝通
免費(fèi)領(lǐng)取
今日已有369人領(lǐng)取成功
劉同學(xué) 138****2860 剛剛成功領(lǐng)取
王同學(xué) 131****2015 剛剛成功領(lǐng)取
張同學(xué) 133****4652 剛剛成功領(lǐng)取
李同學(xué) 135****8607 剛剛成功領(lǐng)取
楊同學(xué) 132****5667 剛剛成功領(lǐng)取
岳同學(xué) 134****6652 剛剛成功領(lǐng)取
梁同學(xué) 157****2950 剛剛成功領(lǐng)取
劉同學(xué) 189****1015 剛剛成功領(lǐng)取
張同學(xué) 155****4678 剛剛成功領(lǐng)取
鄒同學(xué) 139****2907 剛剛成功領(lǐng)取
董同學(xué) 138****2867 剛剛成功領(lǐng)取
周同學(xué) 136****3602 剛剛成功領(lǐng)取
相關(guān)推薦HOT
久久亚洲中文字幕精品一区四,亚洲日本另类欧美一区二区,久久久久久久这里只有免费费精品,高清国产激情视频在线观看
中文日韩欧免费精品视频 | 亚洲国产AⅤ精品一区二区久久 | 视频区国产亚洲.欧美 | 日韩亚洲欧美在线观看 | 亚洲乱码AⅤ日本电影 | 色综合天天综合网在线观看 |