主頁 > 後端開發 > < Python全景系列-9 > Python 裝飾器:優雅地增強你的函式和類

< Python全景系列-9 > Python 裝飾器:優雅地增強你的函式和類

2023-06-01 07:39:33 後端開發

歡迎來到我們的系列博客《Python全景系列》第九篇!在這個系列中,我們將帶領你從Python的基礎知識開始,一步步深入到高級話題,幫助你掌握這門強大而靈活的編程語法,無論你是編程新手,還是有一定基礎的開發者,這個系列都將提供你需要的知識和技能,

** 裝飾器在 Python 中扮演了重要的角色,這是一種精巧的語言特性,讓我們能夠修改或增強函式和類的行為,無需修改它們的源代碼,這篇文章將深入探討裝飾器的所有相關主題,包括裝飾器的基礎知識、實作與使用、作業原理,以及通過實際例子學習裝飾器的獨特用法,**

Python 裝飾器深入探討

在 Python 中,裝飾器提供了一種簡潔的方式,用來修改或增強函式和類的行為,裝飾器在語法上表現為一個前置于函式或類定義之前的特殊標記:

@simple_decorator
def hello_world():
    print("Hello, world!")

在這個例子中,simple_decorator 是一個裝飾器,它作用于下方的 hello_world 函式,裝飾器在概念上就像一個包裝器,它可以在被裝飾的函式執行前后插入任意的代碼,進而改變被裝飾函式的行為,

引數化裝飾器

我們還可以進一步將裝飾器引數化,這讓裝飾器的行為更具靈活性,比如,我們可以定義一個裝飾器,讓它在函式執行前后列印自定義的訊息:

def message_decorator(before_message, after_message):
    def decorator(func):
        def wrapper(*args, **kwargs):
            print(before_message)
            result = func(*args, **kwargs)
            print(after_message)
            return result
        return wrapper
    return decorator

@message_decorator("Start", "End")
def hello_world():
    print("Hello, world!")

在這個例子中,message_decorator 是一個引數化裝飾器,它接受兩個引數,分別代表函式執行前后要列印的訊息,

理解裝飾器的作業原理

在 Python 中,函式是第一類物件,這意味著函式和其他物件一樣,可以作為變數進行賦值,可以作為引數傳給其他函式,可以作為其他函式的回傳值,甚至可以在一個函式里面定義另一個函式,這個特性是實作裝飾器的基礎,

def decorator(func):
    def wrapper():
        print('Before function execution')
        func()
        print('After function execution')
    return wrapper

def hello_world():
    print('Hello, world!')

decorated_hello = decorator(hello_world)
decorated_hello()

在這個例子中,decorator 函式接收一個函式 hello_world 作為引數,并回傳了一個新的函式 wrapped_func,這個新函式在 hello_world 函式執行前后分別列印一條訊息,我們可以看到,裝飾器實際上是一個回傳函式的函式,

函式簽名保持

默認情況下,裝飾器會“掩蓋”掉原函式的名字和檔案字串,這是因為在裝飾器內部,我們回傳了一個全新的函式,我們可以使用 functools.wraps 來解決這個問題:

import functools

def decorator(func):
    @functools.wraps(func)
    def wrapper():
        print('Before function execution')
        func()
        print('After function execution')
    return wrapper

@decorator
def hello_world():
    "Prints 'Hello, world!'"
    print('Hello, world!')

print(hello_world.__name__)
print(hello_world.__doc__)

這樣,使用裝飾器后的函式名和檔案字串能夠保持不變,

Python 裝飾器的應用實體

裝飾器在實際的 Python 編程中有許多應用場景,比如日志記錄、性能測驗、事務處理、快取、權限校驗等,

一個常見的應用就是使用裝飾器進行日志記錄:

import logging

def log_decorator(func):
    logging.basicConfig(level=logging.INFO)
    
    def wrapper(*args, **kwargs):
        logging.info(f'Running "{func.__name__}" with arguments {args} and kwargs {kwargs}')
        result = func(*args, **kwargs)
        logging.info(f'Finished "{func.__name__}" with result {result}')
        return result
    
    return wrapper

@log_decorator
def add(x, y):
    return x + y

這個裝飾器記錄了函式的名稱,函式呼叫的引數,以及函式回傳的結果,

裝飾器鏈

Python 允許我們將多個裝飾器應用到一個函式上,形成一個裝飾器鏈,例如,我們可以同時應用日志裝飾器和性能測驗裝飾器:

import time
import logging
from functools import wraps

def log_decorator(func):
    logging.basicConfig(level=logging.INFO)
    
    @wraps(func)
    def wrapper(*args, **kwargs):
        logging.info(f'Running "{func.__name__}" with arguments {args} and kwargs {kwargs}')
        result = func(*args, **kwargs)
        logging.info(f'Finished "{func.__name__}" with result {result}')
        return result

    return wrapper

def timer_decorator(func):
    @wraps(func)
    def wrapper(*args, **kwargs):
        start_time = time.time()
        result = func(*args, **kwargs)
        end_time = time.time()
        print(f'Function "{func.__name__}" took {end_time - start_time} seconds to run.')
        return result

    return wrapper

@log_decorator
@timer_decorator
def add(x, y):
    time.sleep(2)
    return x + y

在這個例子中,@log_decorator@timer_decorator 兩個裝飾器被同時應用到 add 函式上,它們分別負責記錄日志和測量函式運行時間,

One More Thing: 自動注冊裝飾器

一個有趣的裝飾器應用是自動注冊,這個裝飾器會在裝飾函式時自動將函式添加到一個串列或字典中,這樣我們就可以在程式的其他地方訪問到這個串列或字典,知道有哪些函式被裝飾過,

# 裝飾器將函式注冊到一個串列中
def register_decorator(func_list):
    def decorator(func):
        func_list.append(func)
        return func
    return decorator

# 自動注冊函式
registered_functions = []
@register_decorator(registered_functions)
def foo():
    pass

@register_decorator(registered_functions)
def bar():
    pass

print(registered_functions)  # 輸出: [<function foo at 0x10d38d160>, <function bar at 0x10d38d1f0>]

這個裝飾器可以用于自動注冊路由、插件系統、命令列引數處理等場景,能夠大大提高代碼的靈活性和可擴展性,

總結

Python 裝飾器是一種強大的工具,它可以讓我們更有效地管理和組織代碼,希望通過這篇文章,你能夠更深入地理解裝飾器的作業原理和用法,從而在你的專案中更好地使用裝飾器,

如有幫助,請多關注
個人微信公眾號:【Python全視角】
TeahLead_KrisChang,10+年的互聯網和人工智能從業經驗,10年+技術和業務團隊管理經驗,同濟軟體工程本科,復旦工程管理碩士,阿里云認證云服務資深架構師,上億營收AI產品業務負責人,

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/553941.html

標籤:Python

上一篇:Java中序列化和反序列化解釋

下一篇:返回列表

標籤雲
其他(160069) Python(38190) JavaScript(25469) Java(18172) C(15235) 區塊鏈(8268) C#(7972) AI(7469) 爪哇(7425) MySQL(7219) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5873) 数组(5741) R(5409) Linux(5344) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4580) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2434) ASP.NET(2403) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) .NET技术(1979) 功能(1967) Web開發(1951) HtmlCss(1950) C++(1928) python-3.x(1918) 弹簧靴(1913) xml(1889) PostgreSQL(1879) .NETCore(1863) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • < Python全景系列-9 > Python 裝飾器:優雅地增強你的函式和類

    裝飾器在 Python 中扮演了重要的角色,這是一種精巧的語言特性,讓我們能夠修改或增強函式和類的行為,無需修改它們的源代碼。這篇文章將深入探討裝飾器的所有相關主題,包括裝飾器的基礎知識、實作與使用、作業原理,以及通過實際例子學習裝飾器的獨特用法。 ......

    uj5u.com 2023-06-01 07:39:33 more
  • Java中序列化和反序列化解釋

    在Java中,序列化(Serialization)是指將物件的狀態轉換為位元組流的程序,以便將其保存到檔案、在網路中傳輸或持久化到資料庫中。而反序列化(Deserialization)則是將位元組流轉換回物件的程序,恢復物件的狀態。 序列化和反序列化主要用于以下場景: 1. 物件持久化:通過序列化,可以 ......

    uj5u.com 2023-06-01 07:39:08 more
  • 吃透Redis面試八股文

    > 內容摘自我的學習網站:topjavaer.cn Redis連環40問,絕對夠全! ## Redis是什么? Redis(`Remote Dictionary Server`)是一個使用 C 語言撰寫的,高性能非關系型的鍵值對資料庫。與傳統資料庫不同的是,Redis 的資料是存在記憶體中的,所以讀寫 ......

    uj5u.com 2023-06-01 07:38:56 more
  • 《面試1v1》synchronized

    > 原始碼都背下來了,你給我看這 **我是 javapub,一名 `Markdown` 程式員從👨?💻,八股文種子選手。** **面試官: 你好,我看到你的簡歷上寫著你熟悉 Java 中的 "synchronized" 關鍵字。你能給我講講它的作用嗎?** **候選人:** 當然,"synchro ......

    uj5u.com 2023-06-01 07:38:47 more
  • 單例模式

    ## 單例模式(反射破壞-列舉) #### 餓漢式單例 ~~~java package com.jan.single; //餓漢式單例 public class Hungry { //一上來就會加載好,可能會浪費空間 private byte[] data11=new byte[1024*1024] ......

    uj5u.com 2023-06-01 07:38:30 more
  • 【Netty實戰】1~3章學習筆記

    # 1. Netty總體結構 ## 1.1 Netty簡介 ? Netty是一款用于創建高性能網路應用程式的高級框架。它的基于 Java NIO 的異步的和事件驅動的實作,保證了高負載下應用程式性能的最大化和可伸縮性。 ? 其次,Netty 也包含了一組**設計模式**,將應用程式邏輯從網路層解耦, ......

    uj5u.com 2023-06-01 07:38:10 more
  • Spring MVC官方檔案學習筆記(二)之DispatcherServlet

    **1.DispatcherServlet入門** (1) Spring MVC是以前端控制器模式(即圍繞著一個中央的Servelt, DispatcherServlet)進行設計的,這個DispatcherServlet為請求的處理提供了一個共用的演算法,即它都會將實際的請求處理作業委托給那些可配置 ......

    uj5u.com 2023-06-01 07:37:57 more
  • 別再滿屏找日志了!推薦一款 IDEA 日志管理插件,看日志輕松多了!

    ![](https://img2023.cnblogs.com/other/1218593/202305/1218593-20230531145646615-374710580.png) ## **1.簡介** Grep Console是一款方便開發者對idea控制臺輸出日志進行個性化管理的插件。 ......

    uj5u.com 2023-06-01 07:36:36 more
  • StampedLock:高并發場景下一種比讀寫鎖更快的鎖

    摘要:在讀多寫少的環境中,有沒有一種比ReadWriteLock更快的鎖呢?有,那就是JDK1.8中新增的StampedLock! 本文分享自華為云社區《【高并發】高并發場景下一種比讀寫鎖更快的鎖》,作者: 冰 河。 什么是StampedLock? ReadWriteLock鎖允許多個執行緒同時讀取共 ......

    uj5u.com 2023-06-01 07:31:08 more
  • BIO、NIO、AIO區別詳解

    ###BIO:同步阻塞 主執行緒發起io請求后,需要等待當前io操作完成,才能繼續執行。 ###NIO:同步非阻塞 引入selector、channel、等概念,當主執行緒發起io請求后,輪詢的查看系統是否準備好執行io操作,沒有準備好則主執行緒不會阻塞會繼續執行,準備好主執行緒會阻塞等待io操作完成。 # ......

    uj5u.com 2023-06-01 07:30:33 more