Python day 6(5) Python 函數式編程3

来源:https://www.cnblogs.com/woshihuihui/archive/2018/01/06/8215118.html
-Advertisement-
Play Games

一:裝飾器 1 函數對象有一個__name__屬性,可以拿到函數的名字 上面的log,因為它是一個decorator,所以接受一個函數作為參數,並返回一個函數。我們要藉助Python的@語法,把decorator置於函數的定義處: 調用now()函數,不僅會運行now()函數本身,還會在運行now( ...


一:裝飾器

1 函數對象有一個__name__屬性,可以拿到函數的名字

>>> def now():
...     print('2015-3-25')
...
>>> f = now
>>> f()
2015-3-25


>>> now.__name__
'now'
>>> f.__name__
'now'
2 要增強now()函數的功能,比如,在函數調用前後自動列印日誌,但又不希望修改now()函數的定義,這種在代碼運行期間動態增加功能的方式,稱之為“裝飾器”(Decorator)。本質上,decorator就是一個返回函數的高階函數。
def log(func):
    def wrapper(*args, **kw):
        print('call %s():' % func.__name__)
        return func(*args, **kw)
    return wrapper

上面的log,因為它是一個decorator,所以接受一個函數作為參數,並返回一個函數。我們要藉助Python的@語法,把decorator置於函數的定義處:

@log
def now():
    print('2015-3-25')

調用now()函數,不僅會運行now()函數本身,還會在運行now()函數前列印一行日誌:

>>> now()
call now():
2015-3-25

@log放到now()函數的定義處,相當於執行了語句:

now = log(now)

由於log()是一個decorator,返回一個函數,所以,原來的now()函數仍然存在,只是現在同名的now變數指向了新的函數,於是調用now()將執行新函數,即在log()函數中返回的wrapper()函數。

wrapper()函數的參數定義是(*args, **kw),因此,wrapper()函數可以接受任意參數的調用。在wrapper()函數內,首先列印日誌,再緊接著調用原始函數。

3  若decorator本身需要傳入參數,那就需要編寫一個返回decorator的高階函數,寫出來會更複雜。

def log(text):
    def decorator(func):
        def wrapper(*args, **kw):
            print('%s %s():' % (text, func.__name__))
            return func(*args, **kw)
        return wrapper
    return decorator
@log('execute')
def now():
    print('2015-3-25')

執行結果如下:

>>> now()
execute now():
2015-3-25

和兩層嵌套的decorator相比,3層嵌套的效果是這樣的:

>>> now = log('execute')(now)

我們來剖析上面的語句,首先執行log('execute'),返回的是decorator函數,再調用返回的函數,參數是now函數,返回值最終是wrapper函數。

4   以上兩種decorator的定義都沒有問題,但還差最後一步。因為我們講了函數也是對象,它有__name__等屬性,但你去看經過decorator裝飾之後的函數,它們的__name__已經從原來的'now'變成了'wrapper'

>>> now.__name__
'wrapper'

因為返回的那個wrapper()函數名字就是'wrapper',所以,需要把原始函數的__name__等屬性複製到wrapper()函數中,否則,有些依賴函數簽名的代碼執行就會出錯。

Python內置的functools.wraps可以解決這個問題。記住:

a  在定義函數前導入functools模塊。import functools

b 在定義wrapper()的前面加上@functools.wraps(func)即可。

 

二:偏函數(functools模塊中其中一個功能)

1 在介紹函數參數的時候,通過設定參數的預設值,可以降低函數調用的難度。而偏函數也可以做到這一點。

int()函數可以把字元串轉換為整數,當僅傳入字元串時,int()函數預設按十進位轉換:

>>> int('12345')
12345

int()函數還提供額外的base參數,預設值為10。如果傳入base參數,就可以做N進位的轉換:

>>> int('12345', base=8)
5349   #把字元串當做8進位,返回相對應的十進位。
>>> int('12345', 16)
74565  #把字元串當做16進位,返回相對應的十進位。

>>> import functools
>>> int2 = functools.partial(int, base=2)
>>> int2('1000000')
64
>>> int2('1010101')
85


簡單總結functools.partial的作用就是,把一個函數的某些參數給固定住(也就是設置預設值),返回一個新的函數,調用這個新函數會更簡單。
當函數的參數個數太多,需要簡化時,使用functools.partial可以創建一個新的函數,這個新函數可以固定住原函數的部分參數,從而在調用時更簡單。
 
 
 
 


 

 

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

-Advertisement-
Play Games
更多相關文章
  • 前言 本人在通過《C語言程式設計:現代方法(第2版)》自學C語言時,發現國內並沒有該書完整的課後習題答案,所以就想把自己在學習過程中所做出的答案分享出來,以供大家參考。這些答案是本人自己解答,並參考GitHub上相關的分享和Chegg.com相關資料。因為並沒有權威的答案來源,所以可能會存在錯誤的地 ...
  • 用以前學過的知識,可以簡單地做一個超市庫存管理系統: 定義一個商品類: 然後: ...
  • 首先是打開Content Assistant,自動代碼補全 Window Preferences Java Editor Content Assist,在最下麵的Auto Activation區域的Auto activation triggers for java裡面把26個英文字母都敲一遍 CTR ...
  • 定時任務總會遇到任務重疊執行的情況,比如一個任務1分鐘執行一次,而任務的執行時間超過了1分鐘,這樣就會有兩個相同任務併發執行了。有時候我們是允許這種情況的發生的,比如任務執行的代碼是冪等的,而有時候我們可能考慮到一些情況是不允許這種事情發生的。 在實際場景中,我們定時任務調度使用quartz來實現觸 ...
  • 凱魯嘎吉 - 博客園 http://www.cnblogs.com/kailugaji/ 某糖果廠用原料A、B和C按不向比率混合加工而成甲、乙、丙三種糖果(假設混合加工中不損耗原料)。原料A、B、C在糖果甲、乙、丙中的含量、原料成本、加工成本、原料限量及糖果售價如表所示。 問該廠對這三種糖果各生產多 ...
  • 利用Java 8中新引入的LocalDate類來計算時間間隔,本文將用一段極其簡單的代碼來示範如何計算兩個日期之間間隔的年數。 ...
  • 凱魯嘎吉 - 博客園 http://www.cnblogs.com/kailugaji/ 說明: Lingo版本: 某工廠明年根據合同,每個季度末向銷售公司提供產品,有關信息如下表。若當季生產的產品過多,季末有積餘,則一個季度每積壓一噸產品需支付存貯費O.2萬元。現該廠考慮明年的最佳生產方案,使該廠 ...
  • 微信跳一跳輔助工具 準備工具 adb驅動 安卓手機 打開手機的調試模式 usb接好手機和電腦 PyCharm:全宇宙唯一一款專門用於Python開發IDE工具 實現原理: 獲取手機的實時的截圖 點擊起始位置和落地位置 技算兩個點的距離 計算按壓時間 發送按壓指令 重新刷新手機截圖 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...