掌握Python 裝飾器,其實只需要一盞茶的功夫

来源:https://www.cnblogs.com/astropeak/archive/2018/05/12/9029104.html
-Advertisement-
Play Games

裝飾器的語法為 @dec_name ,置於函數定義之前。如: import atexit @atexit.register def goodbye(): print('Goodbye!') print('Script end here') atexit.register 是一個裝飾器,它的作用是將被 ...


裝飾器的語法為 @dec_name ,置於函數定義之前。如:

import atexit

@atexit.register
def goodbye():
  print('Goodbye!')

print('Script end here')

atexit.register 是一個裝飾器,它的作用是將被裝飾的函數註冊為在程式結束時執行。函數 goodbye 是被裝飾的函數。

程式的運行結果是:

Script end here
Goodbye!

可見函數 goodbye 在程式結束後被自動調用。

另一個常見的例子是 @property ,裝飾類的成員函數,將其轉換為一個描述符。

class Foo:
  @property
  def attr(self):
    print('attr called')
    return 'attr value'

foo = Foo()

等價語法

語句塊

@atexit.register
def goodbye():
  print('Goodbye!')

等價於

def goodbye():
  print('Goodbye!')
goodbye = atexit.register(goodbye)

這兩種寫法在作用上完全等價。

從第二種寫法,更容易看出裝飾器的原理。裝飾器實際上是一個函數(或callable),其輸入、返回值為:

 說明示例中的對應
輸入 被裝飾的函數 goodbye
返回值 變換後的函數或任意對象  

返回值會被賦值給原來指向輸入函數的變數,如示例中的 goodbye 。此時變數 goodbye 將指向裝飾器的返回值,而不是原來的函數定義。返回值一般為一個函數,這個函數是在輸入參數函數添加了一些額外操作的版本。

如下麵這個裝飾器對原始函數添加了一個操作:每次調用這個函數時,列印函數的輸入參數及返回值。

def trace(func):
  def wrapper(*args, **kwargs):   1
    print('Enter. Args: %s, kwargs: %s' % (args, kwargs))   2
    rv = func(*args, **kwargs)   3
    print('Exit. Return value: %s' % (rv))   4
    return rv

  return wrapper

@trace
def area(height, width):
  print('area called')
  return height * width

area(2, 3)   5
  1. 1 :定義一個新函數,這個函數將作為裝飾器的返回值,來替換原函數
  2. 2, 4 : 列印輸入參數、返回值。這是這個裝飾器所定義的操作
  3. 3 :調用原函數
  4. 5 :此時 area 實際上是 1 處定義的 wrapper 函數

程式的運行結果為:

Enter. Args: (2, 3), kwargs: {}
area called
Exit. Return value: 6

如果不使用裝飾器,則必須將以上列印輸入參數及返回值的語句直接寫在 area 函數里,如:

def area(height, width):
  print('Enter. Args: %s, %s' % (height, width))
  print('area called')
  rv = height * width
  print('Exit. Return value: %s' % (rv))
  return rv

area(2, 3)

程式的運行結果與使用裝飾器時相同。但使用裝飾器的好處為:

  1. 列印輸入參數及返回值這個操作可被重用

    如對於一個新的函數 foo ,裝飾器 trace 可以直接拿來使用,而無須在函數內部重覆寫兩條 print 語句。

    @trace
    def foo(val):
      return 'return value'
    

    一個裝飾器實際上定義了一種可重覆使用的操作

  2. 函數的功能更單純

    area 函數的功能是計算面積,而調試語句與其功能無關。使用裝飾器可以將與函數功能無關的語句提取出來。 因此函數可以寫地更小。

    使用裝飾器,相當於將兩個小函數組合起來,組成功能更強大的函數

修正名稱

以上例子中有一個缺陷,函數 areatrace 裝飾後,其名稱變為 wrapper ,而非 areaprint(area) 的結果為:

<function wrapper at 0x10df45668>

wrapper 這個名稱來源於 trace 中定義的 wrapper 函數。

可以通過 functools.wraps 來修正這個問題。

from functools import wraps 

def trace(func):
  @wraps(func) 
  def wrapper(*args, **kwargs):
    print('Enter. Args: %s, kwargs: %s' % (args, kwargs))
    rv = func(*args, **kwargs)
    print('Exit. Return value: %s' % (rv))
    return rv

  return wrapper

@trace
def area(height, width):
  print('area called')
  return height * width

即使用 functools.wraps 來裝飾 wrapper 。此時 print(area) 的結果為:

<function area at 0x10e8371b8>

函數的名稱能夠正確顯示。

接收參數

以上例子中 trace 這個裝飾器在使用時不接受參數。如果想傳入參數,如傳入被裝飾函數的名稱,可以這麼做:

from functools import wraps

def trace(name):
  def wrapper(func):
    @wraps(func)
    def wrapped(*args, **kwargs):
      print('Enter %s. Args: %s, kwargs: %s' % (name, args, kwargs))
      rv = func(*args, **kwargs)
      print('Exit %s. Return value: %s' % (name, rv))
      return rv

    return wrapped
  return wrapper

@trace('area')
def area(height, width):
  print('area called')
  return height * width

area(2, 3)

程式的運行結果為:

Enter area. Args: (2, 3), kwargs: {}
area called
Exit area. Return value: 6

將函數名稱傳入後,在日誌同時列印出函數名,日誌更加清晰。

@trace('area') 是如何工作的?

這裡其實包含了兩個步驟。 @trace('area') 等價於:

dec = trace('area')
@dec
def area(height, width): ...

即先觸發函數調用 trace('area') ,得到一個返回值,這個返回值為 wrapper 函數。 而這個函數才是真正的裝飾器,然後使用這個裝飾器裝飾函數。

多重裝飾器

裝飾器可以疊加使用,如:

@dec1
@dec2
def foo():pass

等價的代碼為:

def foo():pass
foo = dec2(foo)
foo = dec1(foo)

即裝飾器依次裝飾函數,靠近函數定義的裝飾器優先。相當於串聯起來。

如果你一路讀到這裡,我相信你已經掌握了關於Python 裝飾器80%的知識,並能夠應用到工作學習中。你知道了裝飾器的工作原理,以及如何自己編寫一個裝飾器,及避免常見的編寫錯誤。

如果你仍然有疑問,歡迎留言討論!(如果你覺得這篇文章有用,請點下方推薦按鈕:p)

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 下載安裝CMake 下載Opencv源碼 打開CMake,設置源碼路徑和生成路徑,點擊Configure選擇要生成的版本。 勾選BUILD_opencv_world的話,表示只生成一個庫文件(集合起來)。點擊Generate開始生成工程項目。 點擊Open Project,打開工程。生成INSTAL ...
  • _pypiserver_ 是一個最基本的PyPI伺服器實現, 可以用來上傳和維護python包. 本文介紹 _pypiserver_ 在ubuntu上的基本安裝, 配置和使用. 1. 基本安裝和使用 1.1 安裝和啟動 _pypiserver_ 可以在Python 2或者Python 3下運行. 使 ...
  • Layui 官網說這是款經典模塊化前端框架 個人覺得Layui很好用,容易上手。 在學習Layui的之前。先去官網下載必要的文件 將這些文件放入項目當中 然後可以到官網看一下示例。 可以做一個簡單的表格 1 <!DOCTYPE html> 2 <html> 3 <head> 4 <meta char ...
  • 請點擊此處輸入圖片描述 Python下載知乎視頻。 # -*- coding: utf-8 -*- """ 下載知乎視頻: 依賴: pip install requests mac 安裝 ffmpeg: brew install ffmpeg """ import re import uuid im ...
  • 1.在file(文件)裡面找到setting(設置) 2. 然後再左面Editor裡面找Font,再到右面Size裡面設置字體大小 ...
  • 1、String s = new String("xyz");創建了幾個StringObject?是否可以繼承String類? 兩個或一個都有可能,”xyz”對應一個對象,這個對象放在字元串常量緩衝區,常量”xyz”不管出現多少遍,都是緩衝區中的那一個。NewString每寫一遍,就創建一個新的對象 ...
  • 目前想從事分散式服務中間件開發,包括Dubbo,MQ,Redis等,當然包括分散式存儲系統。 ...
  • 視頻 dubbo視頻 spring視頻 springcloud視頻 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...