Python 中如何編寫類型提示

来源:https://www.cnblogs.com/edisonfish/archive/2023/12/18/17911868.html
-Advertisement-
Play Games

哈嘍大家好,我是鹹魚 我們知道 Python 是一門具有動態特性的語言,在編寫 Python 代碼的時候不需要顯式地指定變數的類型 這樣做雖然方便,但是降低了代碼的可閱讀性,在後期 review 代碼的時候容易對變數的類型產生混淆,需要查閱大量上下文,導致後期維護困難 為了提高代碼的可讀性、可維護性 ...


哈嘍大家好,我是鹹魚

我們知道 Python 是一門具有動態特性的語言,在編寫 Python 代碼的時候不需要顯式地指定變數的類型

這樣做雖然方便,但是降低了代碼的可閱讀性,在後期 review 代碼的時候容易對變數的類型產生混淆,需要查閱大量上下文,導致後期維護困難

為了提高代碼的可讀性、可維護性,Python 在 PEP 484 中引入了類型提示( type hinting)。類型提示是 Python 中一個可選但非常有用的功能,可以使代碼更易於閱讀和調試

關於類型提示的介紹可以看:

https://realpython.com/python-type-hints-multiple-types/#use-pythons-type-hints-for-one-piece-of-data-of-alternative-types

在編寫函數的時候,我們通常指定其返回值是一種數據類型,但是在下麵這些情況下可以指定返回不同類型的數據:

  • 當函數使用條件語句返回不同類型結果時
  • 函數有時返回值,有時不返回值
  • 當函數遇到錯誤時,可能需要返回與正常結果的返回類型不同的特定錯誤對象
  • 想要設計更靈活更通用的代碼

那麼這時候該如何編寫類型提示呢?

為常規函數編寫類型提示

def parse_email(email_address: str) -> str | None:
    if "@" in email_address:
        username, domain = email_address.split("@")
        return username
    return None

上面的函數中有一個條件判斷語句,用於檢查參數 email_address 電子郵箱地址裡面是否包含 @ 符號。如果有,則返回用戶名 username ,沒有則返回 None,表示電子郵箱地址不完整

所以該函數的返回值要麼是包含用戶名的字元串,要麼是 None。那麼我們可以用管道符(|) 來表示函數返回單個值的可選類型

# 要麼返回 str ,要麼返回 None
str | None:

在 Python 3.10 之前,我們還可以使用 typing 模塊中的 Union 來表示函數返回的是str 還是 None

from typing import Union

def parse_email(email_address: str) -> Union[str, None]:
    if "@" in email_address:
        username, domain = email_address.split("@")
        return username
    return None

那如果單個返回值裡面包含多個對象的話,該如何編寫類型提示呢?

比如說上面的函數,我希望它:

  • 如果是有效的郵箱,則返回用戶名和功能變數名稱
  • 如果不是有效的郵箱,返回 None

PS: 當返回值里有多個對象時,預設是以元組的形式返回

所以我們可以這麼寫類型提示

def parse_email(email_address: str) -> tuple[str, str] | None:
    if "@" in email_address:
        username, domain = email_address.split("@")
        return username, domain
    return None

tuple[str, str]| None ,表示返回值可以是兩個字元串的元組或None

如果使用 typing 模塊中的 Union來編寫類型提示的話,如下

from typing import Tuple, Union

def parse_email(email_address: str) -> Union[Tuple[str, str], None]:
    if "@" in email_address:
        username, domain = email_address.split("@")
        return username, domain
    return None

舉三反一一下,如果單個返回值包含三個對象,可以這麼寫

# 函數返回值裡面包含了字元串、整數、布爾值
def get_user_info(user: User) -> tuple[str, int, bool]:
    ...

為回調函數編寫類型提示

在 Python 中,函數可以作為另一個函數的參數或者返回其他函數。這種函數被稱為高階函數

比如說 Python內置函數(例如sorted()map()filter())可以接受一個函數作為參數

這個作為參數傳遞的函數通常被稱為回調函數(callback function),因為它在另一個函數中被調用("回調"),回調函數是一種可調用對象(callable objects)

可調用對象指的是可以像函數一樣調用的對象。Python 中可調用對象包括常規函數、lambda 表達式或實現了__call__()方法的類)

那麼我們在調用回調函數的時候,該如何編寫類型註釋呢?

比如說下麵的例子

>>> from collections.abc import Callable

>>> def apply_func(
...     func: Callable[[str], tuple[str, str]], value: str
... ) -> tuple[str, str]:
...     return func(value)
...
>>> def parse_email(email_address: str) -> tuple[str, str]:
...     if "@" in email_address:
...         username, domain = email_address.split("@")
...         return username, domain
...     return "", ""
...
>>> apply_func(parse_email, "[email protected]")
('claudia', 'realpython.com')

在函數 apply_func 的類型提示中,將回調函數 func作為第一個參數,將字元串 value 作為第二個參數,返回值是一個包含兩個 str 的 tuple

Callable[[str], tuple[str, str]]:表示回調函數 func 接收參數是一個 str,返回值是一個包含兩個 str 的 tuple

在函數 parse_email 的類型提示中,接受一個 str 類型的參數 email_address ,返回值類型是一個包含兩個 str 的 tuple

那如果我希望函數 apply_func 能夠接收具有多種輸入類型的不同函數作為參數(比如說回調函數有多個輸入參數)並有多種返回類型,該怎麼辦?

我們可以用省略號... 來表示可調用對象(例如回調函數)可以接受多個參數,這樣就不需要依次列出接受參數的類型

def apply_func( 
	func: Callable[...,tuple[str, str]], value: str) -> tuple[str, str]:
	return func(value)

或者使用 typing 模塊中的類型來指定任何返回 Any 類型

from collections.abc import Callable
from typing import Any

def apply_func( 
	func: Callable[...,Any], *args: Any, **kwargs: Any) -> tuple[str, str]:
	 return func(*args, **kwargs)

我們還可以在類型提示中把回調函數的返回值類型寫成 T ,這是一個類型變數type variable,可以代表任何類型

from collections.abc import Callable
from typing import Any, TypeVar

T = TypeVar("T")

def apply_func(func: Callable[..., T], *args: Any, **kwargs: Any) -> T:
    return func(*args, **kwargs)

apply_func 的返回值類型也是 T,*args: Any, **kwargs: Any 表示 apply_func 可以接受任意數量的參數(包括 0)

為生成器編寫類型提示

在 Python 中,生成器(Generators)是一種特殊的迭代器,它們允許按需生成值,而無需提前生成所有值並將其存儲在記憶體中

生成器逐個產生並返回值,這對於處理大量數據或無限序列非常有用

生成器可以通過函數與 yield 語句創建。yield 語句在生成器函數內部被用來產生一個值,併在暫停生成器的同時返回該值給調用者

每次調用生成器的 next()方法或使用 for迴圈時,生成器函數會從上一次yield語句的位置恢復執行,並繼續執行到下一個yield語句或函數結束

繼續上面的例子,我現在有大量的郵箱需要判斷是否有效,與其將每個解析的結果存儲在記憶體中並讓函數一次返回所有內容,不如使用生成器一次生成一個解析結果

>>> from collections.abc import Generator

>>> def parse_email() -> Generator[tuple[str, str], str, str]:
		# 定義初始的 sent 值為元組 ("", "")
...     sent = yield ("", "")
...     while sent != "":
...         if "@" in sent:
...             username, domain = sent.split("@")
...             sent = yield username, domain
...         else:
...             sent = yield "invalid email"
...     return "Done"

Generator[tuple[str, str], str, str]類型提示裡面有三個參數(後面兩個是可選的),其中:

  • yield 類型:第一個參數是生成器生成的結果。例子中它是一個元組,包含兩個字元串,一個表示用戶名,另一個表示功能變數名稱
  • send 類型:第二個參數表示使用 send 方法發送給生成器的內容。例子中是一個字元串,表示發送的郵箱地址
  • return 類型:第三個參數表示生成器生成值後返回的內容。例子中函數返回字元串“Done”

然後調用該生成器

>>> generator = parse_email()
>>> next(generator)
('', '')
#使用 send 方法向生成器發送參數
>>> generator.send("[email protected]")
('claudia', 'realpython.com')
>>> generator.send("realpython")
'invalid email'
>>> try:
...     generator.send("")
... except StopIteration as ex:
...     print(ex.value)
...
Done

首先調用生成器函數,該函數將返回一個新的 parse_email() 生成器對象。然後,通過調用內置 next() 函數將生成器推進到第一個 yield 語句

之後開始向生成器發送電子郵件地址進行解析。當發送空字元串或不帶 @ 符號的字元串時,生成器將終止

又因為生成器也是迭代器,因此也可以使用 collections.abc.Iterator 而不是 Generator 來進行類型提示

但是如果使用了 collections.abc.Iterator 類型提示,就不能指定 send 類型和 rerurn 類型,因此只有當生成器只生成值時 collections.abc.Iterator 才起作用

from collections.abc import Iterator

def parse_emails(emails: list[str]) -> Iterator[tuple[str, str]]:
    for email in emails:
        if "@" in email:
            username, domain = email.split("@")
            yield username, domain

我們還可以在接收參數裡面使用 Iterable 類型提示,這樣表示函數 parse_emails 可以接受任何可迭代對象,而不僅僅是像以前那樣的列表

from collections.abc import Iterable

def parse_emails(emails: Iterable[str]) -> Iterable[tuple[str, str]]:
    for email in emails:
        if "@" in email:
            username, domain = email.split("@")
            yield username, domain

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

-Advertisement-
Play Games
更多相關文章
  • 大家好,我是 Java陳序員。 這幾天,逛 Github 的時候,看到了一個項目 win12 —— 仿 Windows12 網頁版!被它實現的頁面功能震撼到了,大家可以一起來感受下! 首先是登錄頁面。 登錄後,再看看 “電腦桌面”,這風格完全和 Windows 操作系統的一模一樣!預設的應用程式有 ...
  • 一、定義 義一個對象來封裝一系列的對象交互。中介者模式使各對象不需要顯示地相互引用,從而使其耦合鬆散,而且可以讓你相對獨立地改變它們之間的交互。中介者模式又稱為調停模式,它是一種對象說行為型模式。 二、描述 在中介者模式中,引入了用於協調其他對象/類之間的相互調用的中介者類,為了讓系統具有更好的靈活 ...
  • C-08 聚合(分組)函數和SQL(DQL)執行流程 上一章,講到了SQL單行函數。實際上SQL函數還有一類,叫做聚合函數,或者多行,分組函數,它是對一組數據進行彙總的函數,輸入的是一組數據的集合,輸出的是單個值。 1.聚合函數的介紹 什麼是聚合函數 聚合函數作用於一組數據,並對一組數據返回一個值。 ...
  • 如題,這是一個很簡單也很基礎的問題,先看不同操作的結果有什麼不同 結果1:項目下出現了一個空“文件夾” 結果2:項目下多了一個“Python Package”,該package下包含了一個“__init__.py”文件,該py文件是空的 如果把該py文件刪掉後,package也就變成了directo ...
  • 1 數據可視化 自 DataGrip 2023.3 發佈以來,已整合 Lets-Plot 庫,實現數據可視化。該可視化功能可用於所有三種類型的網格: 主選項卡:在打開表、視圖或 CSV 文件時,在分割模式下顯示圖表。 結果選項卡:在 服務 工具視窗中觀察查詢結果時,可以顯示圖表而不是網格。 編輯器中 ...
  • 在Java中,我們經常需要獲取某個類的Class對象,以便進行反射操作、動態載入類或者獲取類的信息。本文將介紹Java中獲取Class對象的三種常見方式。 1. 使用.class語法 獲取Class對象最簡單的方式就是使用.class語法。每個Java類都有一個隱含的靜態成員變數class,可以通過 ...
  • C 語言中的類型轉換 有時,您必須將一種數據類型的值轉換為另一種類型。這稱為類型轉換 隱式轉換 當您將一種類型的值分配給另一種類型的變數時,編譯器會自動進行隱式轉換。 例如,如果您將一個 int 值分配給一個 float 類型: // 自動轉換:int to float float myFloat ...
  • 為什麼需要熱載入 在某些情況,你可能不希望關閉Python進程並重新打開,或者你無法重新啟動Python,這時候就需要實現實時修改代碼實時生效,而不用重新啟動Python 在我的需求下,這個功能非常重要,我將Python註入到了其他進程,並作為一個線程運行。如果我想關閉Python,要麼殺死Pyth ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...