Python裝飾器實例講解(二)

来源:https://www.cnblogs.com/wuxianfeng023/archive/2023/02/09/17106513.html
-Advertisement-
Play Games

Python裝飾器實例講解(二) Python裝飾器實例講解(一) 你最好去看下第一篇,雖然也不是緊密的鏈接在一起 參考B站碼農高天的視頻,大家喜歡看視頻可以跳轉忽略本文:https://www.bilibili.com/video/BV19U4y1d79C 一鍵三連哦 本文的知識點主要是 ​ 類裝 ...


Python裝飾器實例講解(二)

Python裝飾器實例講解(一)

你最好去看下第一篇,雖然也不是緊密的鏈接在一起

參考B站碼農高天的視頻,大家喜歡看視頻可以跳轉忽略本文:https://www.bilibili.com/video/BV19U4y1d79C
一鍵三連哦
本文的知識點主要是

​ 類裝飾器

​ 裝飾器的本質(up主說的萬能公式)

案例

  • 代碼

    def count_time(func):
        def wrapper(*args,**kwargs):
            from time import time
            start_time = time()
            result = func(*args,**kwargs)
            end_time = time()
            print(f'統計花了{end_time-start_time}時間')
            return result
        return wrapper
    
  • 改造為類裝飾器(註意對比)

    • 你得知道基礎的python的面向對象的知識
    • 一些類的魔術方法如__init__和__call__
    class CountTime:
        def __init__(self,function_name):  # 類沒傳參一說,但實例化是可以傳參的,類比  def count_time(func):
            self.function_name = function_name
        def __call__(self, *args, **kwargs):  # 類實例的(),像函數的call , ==>def wrapper(*args,**kwargs):
            from time import time
            start_time = time()
            result = self.function_name(*args,**kwargs)  # 也就改了這裡,其他都一樣
            end_time = time()
            print(f'統計花了{end_time-start_time}時間')
            return result
    
  • 完整的代碼

    def is_prime(x):
        if  x == 2 :
            return True
        elif x % 2 == 0 or x == 1 :
            return False
        for i in range(3, int(x ** 0.5) + 1, 2):
            if x % i == 0:
                return False
        return True
    
    class CountTime:
        ... # 就不重覆上面了
    
    @CountTime  # 類是一個裝飾器
    def get_prime_nums(start,end):
        prime_nums = 0
        for num in range(start,end):
            if is_prime(num):
                prime_nums = prime_nums + 1
        return prime_nums
    
    
    print(get_prime_nums(2,50000))  # 效果是一樣的
    

碼農高天說

我把up主的一些話摘錄一些寫到這裡,輔助大家理解

  • 裝飾器decorator:是一個輸入是函數,輸出也是函數的函數(看講解一中的裝飾器)
  • 類裝飾器 class decorator,up主說有一定的歧義
    • 可以當做裝飾器的類(裝飾器本身)
    • 可以裝飾類的裝飾器(裝飾器要裝飾的對象)
  • 裝飾器本身既可以是函數也可以是類,裝飾的對象同樣可以是函數或者類
  • 背這些理論沒有意義,關鍵要弄懂背後的原理
  • __call__可以讓類的實例當做函數用(就是callable)

萬能公式

  • 裝飾器語法糖背後

    class CountTime:
        ...# 同上
    
    @CountTime
    def add(a,b):  # 就用碼農的demo函數
        return a+b
    
    
    print(add(1,2))
    
  • @CountTime等價於,所謂的萬能公式

    add = CountTime(add)
    
  • print(add(1,2))已經不再是使用的原始的add了,用的是新的add

    print(add(1,2)) 等價於
    
    print(CountTime(add)(1,2))
    
  • 也就是說

    # @CountTime  # 去掉裝飾器,你就是定義了一個簡單的函數
    # # add = CountTime(add) # 函數名被重定義了  相當於這樣
    def add(a,b):
        return a+b
    
    print(CountTime(add)(1,2))
    
  • 你還可以這樣

    def add(a,b):
        return a+b
    
    
    new_add = CountTime(add)
    print(new_add(1,2))
    
  • 是的,被裝飾過的函數已經不再是原來的函數了,它總是會先去執行裝飾器(CountTime(add))

  • 總結:

    • 在一個函數上做裝飾器,等價於裝飾器調用這個函數
    • 在類裝飾器的這個例子中,add從一個函數變成了一個類的實例(type看下即可)

改造,有參數的裝飾器

  • 我們看到過很多的裝飾是有參數的,這是怎麼實現的呢?

  • 比如你想要輸出的信息可以調整其首碼

    計時: 0.46秒
    	或者
    用時: 0.46秒
    
  • 你希望是這樣裝飾和調用的

    @CountTime(prefix='用時:')
    def add(a,b):
        return a+b
    
    
    print(add(1,2))
    
  • 那咋實現呢?

  • 回到萬能公式:

    @CountTime(prefix='用時:')
    def add(a,b):
        ...
    # 等價於
    add = CountTime(add)
    
    # 那麼
    @CountTime(prefix='用時:')
    def add(a,b):
        ...
        
    # 等價於
    add = CountTime(prefix='用時:')(add)
    
  • CountTime這個類能CountTime(prefix='用時:'),就是實例化做到的,所以...類的init方法要改一下,不再是傳參function_name了,而是傳你的prefix,像這樣

    class CountTimeV2:
        def __init__(self,prefix='用時:'):
            self.prefix = prefix
    
  • 但現在還不能繼續,add = CountTime(prefix='用時:')(add)中你(add)還要處理,前面是init做的,()就是callable做的,裡面的參數是add,也就是函數的名字,所以你的call也要改造,像這樣嗎?

        def __call__(self, function_name):
            from time import time
            start_time = time()
            result = function_name(*args,**kwargs)
            end_time = time()
            print(f'統計花了{end_time-start_time}時間')
            return result
    
  • 不對的,光這樣改造不夠的,因為你這個function_name(*args,**kwargs)在IDE中就會報錯,哪裡來的呢?沒有定義。

  • 回想講解一中,函數裝飾器裡層,還有一個函數,此處就可以參考

    class CountTimeV2:
        def __init__(self, prefix='用時:'):
            self.prefix = prefix
    
        def __call__(self, function_name):
            def wrapper(*args, **kwargs):  # 加了個函數 , 包裹一層
                from time import time
                start_time = time()
                result = function_name(*args, **kwargs) # 這樣就可以用參數了
                end_time = time()
                print(f'{self.prefix}{end_time - start_time}')  # 用之前的定義
                return result
    
            return wrapper
    
    
    @CountTimeV2(prefix='耗時:')  # 可以改為用時、計時等等
    def add(a, b):
        return a + b
    
    
    print(add(1, 2))
    
    

前面談的是類是一個裝飾器,裝了一個函數

下麵談的是函數是一個裝飾器,裝飾一個類

類的裝飾器

  • 現在有這麼一個類

    class Person:
        pass
    
    wuxianfeng = Person()
    print(wuxianfeng)  # <__main__.Person object at 0x000002361C15A460>
    
  • 你學過python可以這樣修改

    class Person:
        def __str__(self):
            return f"{self.__class__.__name__}"
    
    wuxianfeng = Person()
    print(wuxianfeng) # Person
    
  • 但如果有很多的類都要如此呢?

  • 可以寫個裝飾器,來裝飾這些類唄

  • 怎麼寫?回想剛纔你學到的知識,萬能公式!

    def show_classname(): # 先不寫參數
        pass  # 先不寫內容
    
    @show_classname
    class Person:
        pass
    
    Person = show_classname(Person)
    
    wuxianfeng = Person()
    print(wuxianfeng)
    
    • 你現在要寫一個函數,名字隨意,如show_classname

    • 你肯定要裝飾在類上

      @show_classname
      class Person:
          pass
      
    • 根據萬能公式,你的Person應該變了

      Person = show_classname(Person)
      # 從上面這段代碼,你要能分析出以下內容
          # 1. show_classname應該有個參數,傳參是個類名
          # 2. 因為可以Person = ,所以show_classname有個返回值
      
    • 對於使用者而言,應該沒有任何操作上的差異

      wuxianfeng = Person()
      # 從上面這段代碼,你要能分析出以下內容
      # 1. Person已經被你改變了
      # 2. Person()==>show_classname(Person)(),所以show_classname這個函數的返回值還是一個類
      print(wuxianfeng)
      
      
      
    • 分析完了,函數體部分是有點不好理解的

      def show_classname(class_name):
          def __str__(self):
              return self.__class__.__name__
          class_name.__str__ = __str__
          return class_name
      
      @show_classname
      class Person:
          pass
      
      
      Person = show_classname(Person)
      
      wuxianfeng = Person()
      print(wuxianfeng)
      
    • 看著這個結果,我們來解釋下(也許你會更好理解)

      1. show_classname(Person) 返回仍然是Person
      2. 但這個時候的Person被改變了一點(你要做的不就是如此嗎?)
      3. 原來你是這樣寫的
          class Person:
              def __str__(self):
                  return f"{self.__class__.__name__}"
          看看現在的寫法
          def __str__(self):
              return self.__class__.__name__
          class_name.__str__ = __str__ 
               # 前面的class_name.__str__ 是類自己的函數(本段解釋的line 5)
               # 後面的= __str__ ,是line8的函數
               # 是的,函數可以被重新賦值,函數是一等對象,
      
    • 如果還不明白...儘力了

帶參數的類的裝飾器

碼農高天並沒有給出示例代碼

當然如果你真懂了前面的"改造,有參數的裝飾器",也很簡單

  • 直接上代碼

    def show_classname(info='類名:'):
        def wrapper(class_name):
            def __str__(self):
                return info+ self.__class__.__name__
            class_name.__str__ = __str__
            return class_name
        return wrapper
    
    @show_classname('類的名字是:')  #
    class Person:
        pass
    
    wuxianfeng = Person()
    print(wuxianfeng)
    
    
    
    
  • 預設值就是='類名:',怎麼用呢

    @show_classname()
    class Human:
        pass
    qianyuli = Human()
    print(qianyuli)
    
  • 註意不能這樣

    @show_classname
    class Human:
        pass
    qianyuli = Human()
    print(qianyuli)
    
  • 提示錯誤

    Traceback (most recent call last):
      File "demo.py", line 21, in <module>
        qianyuli = Human()
    TypeError: wrapper() missing 1 required positional argument: 'class_name'
    
  • 提個問題,為何會報錯?

  • 如果你無法解釋的通,你應該還沒理解。

  • 答案其實還是萬能公式。

    @show_classname
    class Human:
        pass
    # 1. 
    等價於(萬能公式來了)
    Human = show_classname(Human)
    
    # 2. 
    show_classname(Human) 執行這個的時候其實你在做
    def show_classname(info='類名:'):
        ...
    Human這個東西傳給了info
    # 你要不信,你改為下麵這樣就知道了;信的話就過
                def show_classname(info='類名:'):
                    print('info是啥?',info.name)
                class Human:
                    name = '女媧'
                    
    # 3. 
    show_classname(Human)這個的返回是wrapper
    但wrapper這個函數是有個參數的,看你的定義def  wrapper(class_name):
        
    # 4. 
    定義的時候是感知不到問題的,下麵的報錯行
    qianyuli = Human()
    
    其實你是在
    Human()=>show_classname(Human)()=>wrapper(),錯了,(看3),你需要一個class_name參數
    
    
  • 如果還不明白...儘力了


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

-Advertisement-
Play Games
更多相關文章
  • SpringMVC底層機制簡單實現-01 主要完成:核心分發控制器+Controller和Service註入容器+對象自動裝配+控制器方法獲取參數+視圖解析+返回JSON格式數據 1.搭建開發環境 創建 Maven 項目,File-New-Project-Maven 將 pom.xml 文件中的編譯 ...
  • 在系統開發過程中,一些需要修改的參數和變數通常會從代碼中分離出來,以獨立配置文件的形式進行獨立管理。其目的是使靜態系統工件或交付(如war、jar包等)更好地適應實際的物理運行環境。配置管理一般包括在系統部署過程中,由系統管理員或運維人員完成。配置變更是調整運行系統行為的有效手段之一。Nacos可以... ...
  • [ ] 一、支持的IDE和版本 支持JB家族所有IDE激活 支持版本為2021.3~2022.3.2 二、如何破解激活 第一步:激活工具下載 為了防止破解工具被刪除,通過公眾號回覆“永久激活”獲取下載最新工具(如過期,請記得提醒我哦) 關註公眾號後臺回覆“永久激活”,獲取最新激活工具和教程。 公眾號 ...
  • 這篇文章主要描述在分散式系統下如何實現事務處理,包括三種常見的實現事務的方法:基於XA協議的二階段提交方法、三階段提交方法和基於分散式消息的最終一致性方案。 ...
  • 《Terraform 101 從入門到實踐》這本小冊在南瓜慢說官方網站和GitHub兩個地方同步更新,書中的示例代碼也是放在GitHub上,方便大家參考查看。 不怕出身低,行行出狀元。 插件 Terraform可以對多種平臺的多種資源進行管理,這個是通過插件來實現的。 這裡的插件,在Terrafor ...
  • 一、概述 String類的一個最大特性是不可修改性,而導致其不可修改的原因是在String內部定義了一個常量數組,因此每次對字元串的操作實際上都會另外分配分配一個新的常量數組空間。 二、創建字元串對象的方式 2.1 四種方式 方式一:直接賦值(常用) // 直接賦值方式創建對象是在方法區的常量池 S ...
  • 教程簡介 人工智慧與Python初學者教程 - 通過簡單易學的步驟學習人工智慧從基礎到高級概念,包括入門概念,入門,機器學習,數據準備,監督學習:分類,監督學習:回歸,邏輯編程,無監督學習:聚類,性能考慮,自然語言處理,NLTK包,分析時間序列數據,語音識別,啟髮式搜索,游戲,神經網路,強化學習,遺 ...
  • Rust 語言由 Mozilla 開發,最早發佈於 2014 年 9 月,是一種高效、可靠的通用高級語言。其高效不僅限於開發效率,它的執行效率也是令人稱贊的,是一種少有的兼顧開發效率和執行效率的語言。 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...