主頁 > 後端開發 > Python 型別注解

Python 型別注解

2023-02-13 07:06:47 後端開發

在Python語言發展的程序中,PEP提案發揮了巨大的作用,如PEP 3107 和 PEP 484提案,分別給我們帶來了函式注解(Function Annotations)和型別提示(Type Hints)的功能,

PEP 3107:定義了函式注解的語法,允許為函式的引數和回傳值添加元資料注解,

PEP 484:按照PEP 3107函式注解的語法,從Python語法層面全面支持型別提示,型別提示可以是內置型別、內置類、抽象基類、types模塊中提供的型別和開發人員自定義的類,

另外 PEP 526, PEP 544, PEP 586, PEP 589, PEP 591 這些東西對 PEP 3107 和 PEP 484 進行了補充,比如添加了變數注釋,字面量注釋這些東西,

需要注意的是,型別提示僅有提示的作用,這里的提示是指用戶閱讀Python代碼的時候的提示,僅在語法層面支持,對代碼的運行沒有任何影響,Python 解釋器在運行代碼的時候會忽略型別提示,也就是說,Python的型別提示僅是為了提升代碼可讀性,一定程度上緩解"動態語言一時爽,代碼重構火葬場"的尷尬,

下面將函式注解和型別提示,統稱為型別注解,


型別注解優點

1、可以使Python擁有部分靜態語言的特性,利用型別注解可以實作一種類似型別宣告的效果,提升代碼的可讀性及后續的可維護性,

2、型別注解可以讓IDE(如pycharm)像靜態語言那樣分析我們的代碼,及時給我們相應的提示,如下圖對比:

image-20230212140705227
VS
image-20230212140817448

3、多多使用型別注解,不僅可以讓Python擁有強型別語言的嚴謹,還能保持Python作為動態型別語言的靈活性,


普通變數型別注解

在宣告變數時,變數的后面可以加一個冒號,后面再寫上變數的型別,如 int、list 等等,以此實作型別注解,

a: int = 22
b: str = "name"
c: float = 55.5
d: bool = True
e: list = [1, 2, 3]
f: set = {1, 2, 3}
g: dict = {"name": "ming", "age": 22}
h: tuple = (1, 2, 3)
i: bytes = b'world'
j: bytearray = bytearray("world")

函式引數及回傳值型別

函式引數的型別宣告就是冒號+型別即可,和普通變數型別宣告沒區別,

函式回傳值的型別宣告是用箭頭指向具體的型別,如果是回傳值有多個,使用元組包裹即可(因為函式的多個回傳值就是以元組形式回傳的),需要注意的是,箭頭左右兩邊都要留有空格,

def handler(a: int, b: int) -> int:
    return a + b


def handler2(a: int, b: int, *args: int) -> int:
    return a + b + sum(args)


def handler3(a: int, b: int, *args: int, **kwargs: int) -> (int, str):
    return a + b + sum(args) + sum(kwargs.values()), ""

typing模塊

typing模塊的加入不會影響程式的運行,也不會報正式的錯誤,pycharm支持檢測基于typing注解的錯誤,不符合規定型別注解時會出現黃色警告,但不會影響程式運行,

容器型別 & 復合型別

串列、字典、元組等包含元素的復合型別,用簡單的 list,dict,tuple 不能夠明確說明內部元素的具體型別,

此外,Python本身就是動態型別的語言,如果我們強制使用某種型別,一定程度上會喪失Python作為動態語言的優勢,因此 typing 模塊提供了一種復合型別注解的語法,即一個引數即可以是型別A,也可以是型別B或者型別C

from typing import Dict, List, Set, Tuple, Union

# 字典
d: Dict[str, int] = {"a": 1, "b": 2}
d1: Dict[str, int or str] = {"a": 1, "b": "2"}  	# 使用or表示支持多個型別

# 串列
l: List[int] = [1, 2, 3]
l1: List[int or str] = [1, 2, "3"]

# 元組
t: Tuple[str, int] = ("a", 1)		# 代表了構成元組的第一個元素是 str 型別,第二個元素是 int 型別
t1: Tuple[str, ...] = ("a", "b", "c", "d", "e", "f", "g")		# 代表接受多個 str 型別的元素
t2: Tuple[str or int, ...] = ("a", "b", 2)		# 代表接受多個 str 或 int 型別的元素

# 集合
s: Set[int] = {1, 2, 3, 4}
s1: Set[Union[int, str, float]] = {1, "2", 3.333, 4}	# Union 同 or

TypedDict

TypedDict宣告一個字典型別,該型別期望它的所有實體都有一組固定的keys,其中每個key都與對應型別的值關聯,

from typing import TypedDict


class Student(TypedDict):
    name: str
    age: int
    height: float


s1: Student = {
    "name": "xiao ming",
    "age": 22,
    "height": 55.5
}

s2: Student = {
    "name": "xiao hong",
    "age": 21,
}
image-20230212162843194

可以看出,pycharm也會警告我們字典實體中缺失的key,

同時,在我們生成字典實體的時候,pycharm也會給我們key的提示,

image-20230212163042746

型別別名

型別別名是通過將型別分配給別名來定義的,型別別名可用于簡化復雜型別提示,

from typing import Union

Number = Union[int, float]

def process(v: Number) -> Number:
    return v

x: Number = 2
y: Number = 2.2
process(x)
process(22)		# 型別檢查成功,型別別名和原始型別是等價的

NewType

使用NewType輔助類來創建不同的型別

from typing import NewType

Number = NewType("Number", int)

def process(v: Number) -> Number:
    return v

x: Number = Number(22)
process(x)
process(22)     # 型別檢查例外:Expected type 'Number', got 'int' instead 
# 原因就是NewType創建的是原始型別的“子型別”

因此,型別別名 和 NewType 具體使用哪個,要視情況而定,不知道使用哪個,可以先使用型別別名,


NoReturn

當一個方法沒有回傳結果時,為了注解它的回傳型別,我們可以將其注解為 NoReturn,

因為Python 的函式運行結束時隱式回傳 None ,這和真正的無回傳值是有區別的,

from typing import NoReturn

def process() -> NoReturn:
    pass

可選型別:Optional

使用 Optional[] 表示可能為 None 的值

from typing import Optional

def handler(x: int) -> Optional[int]:
    if x % 2 == 0:
        return x

可呼叫物件:Callable

若一個變數型別是可呼叫函式,則可以用 Callable[[Arg1Type, Arg2Type], ReturnType] 實作型別提示

from typing import Optional, Callable

def handler(x: int) -> Optional[int]:
    if x % 2 == 0:
        return x

def handler2(func: Callable[[int], Optional[int]]):
    pass

handler2(handler)

字面量:Literal

指示相應的變數或函式引數只接收與提供的字面量(或多個字面量之一)等效的值,可以理解為規定了某個引數或變數的所有列舉值,

from typing import Literal, NoReturn

Mode = Literal["r", "w"]


def process(mode: Mode) -> NoReturn:
    pass


process("s")
image-20230212162309699

可以看出,pycharm檢查出了我們輸入的值并不符合字面量規定的值,進而出現了黃色警告,

image-20230212163929671

Any

是一種特殊的型別,每種型別都視為與Any兼容,同樣,Any也與所有型別兼容,可以對Any型別的值執行任何操作或方法呼叫,并將其分配給任何變數,將Any型別的值分配給更精確的型別(more precise type)時,不會執行型別檢查,所有沒有回傳型別或引數型別的函式都將隱式地默認使用Any,

使用Any,說明值是動態型別,

把所有的型別都注解為 Any 將毫無意義,因此 Any 應當盡量少使用

from typing import Any

def foo() -> Any:
    pass

抽象基類

# 在某些情況下,我們可能并不需要嚴格區分一個變數或引數到底是串列 list 型別還是元組 tuple 型別
# 可以使用一個更為泛化的型別,叫做 Sequence,其用法類似于 List
class typing.Sequence(Reversible[T_co], Collection[T_co])


# collections.abc.Iterator的泛型版本
# 注釋函式引數中的迭代型別時,推薦使用的抽象集合型別
class typing.Iterable(Generic[T_co])

def print_iterable(x: Iterable):
    for i in x:
        print(i)



# collections.abc.Mapping的泛型(generic)版本
# 注釋函式引數中的Key-Value型別時,推薦使用的抽象集合型別
class typing.Mapping(Sized, Collection[KT], Generic[VT_co])

泛型:TypeVar

先拋出問題:

假設有一個函式,要求它既能夠處理字串,又能夠處理數字,那么你可能很自然地想到了 Union ,如下:

from typing import Union

AddValue = https://www.cnblogs.com/iorson/archive/2023/02/12/Union[int, str]


def add(a: AddValue, b: AddValue) -> AddValue:
    return a + b


if __name__ =="__main__":
    print(add(1, 2))        # 型別檢查通過,輸出 3
    print(add("1", "2"))    # 型別檢查通過,輸出 12
    print(add("1", 2))      # 型別檢查通過,報錯 TypeError: can only concatenate str (not "int") to str

在型別檢查通過的情況下,我們完成并運行了這段代碼,可是代碼卻報錯了!

原因就是我們的初衷是數字和數字相加實作求和,字串和字串相加實作拼接,沒有考慮到字串與數字混用的問題,從而引發錯誤,

根據以上問題,我們可以引入泛型來解決這個問題:

from typing import TypeVar

AddT = TypeVar("AddT", int, str)


def add(a: AddT, b: AddT) -> AddT:
    return a + b


if __name__ == "__main__":
    print(add(1, 2))		# 型別檢查通過,輸出 3
    print(add("1", "2"))	# 型別檢查通過,輸出 12
    print(add("1", 2))		# 型別檢查失敗,pycharm告警 Expected type 'str' (matched generic type 'AddT'), got 'int' instead

"""
通過告警,我們提前發現了混用型別的問題,避免了程式運行時發生例外的可能,
"""

泛型很巧妙地對型別進行了引數化,同時又保留了函式處理不同型別時的靈活性,


參考

1、Python 標準庫 typing 型別注解標注

2、Python型別注解,你需要知道的都在這里了

本文來自博客園

作者:奧森iorson

轉載請注明原文鏈接:https://www.cnblogs.com/iorson/p/17114352.html

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

標籤:其他

上一篇:Maven基礎學習

下一篇:QA 不講武德!線上 1 億+ 資料亂分頁,讓我搞到半夜。。

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(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
最新发布
  • Rust中的智能指標:Box<T> Rc<T> Arc<T> Cell<T> RefCell<T> Weak

    Rust中的智能指標是什么 智能指標(smart pointers)是一類資料結構,是擁有資料所有權和額外功能的指標。是指標的進一步發展 指標(pointer)是一個包含記憶體地址的變數的通用概念。這個地址參考,或 ” 指向”(points at)一些其 他資料 。參考以 & 符號為標志并借用了他們所 ......

    uj5u.com 2023-04-20 07:24:10 more
  • Java的值傳遞和參考傳遞

    值傳遞不會改變本身,參考傳遞(如果傳遞的值需要實體化到堆里)如果發生修改了會改變本身。 1.基本資料型別都是值傳遞 package com.example.basic; public class Test { public static void main(String[] args) { int ......

    uj5u.com 2023-04-20 07:24:04 more
  • [2]SpinalHDL教程——Scala簡單入門

    第一個 Scala 程式 shell里面輸入 $ scala scala> 1 + 1 res0: Int = 2 scala> println("Hello World!") Hello World! 檔案形式 object HelloWorld { /* 這是我的第一個 Scala 程式 * 以 ......

    uj5u.com 2023-04-20 07:23:58 more
  • 理解函式指標和回呼函式

    理解 函式指標 指向函式的指標。比如: 理解函式指標的偽代碼 void (*p)(int type, char *data); // 定義一個函式指標p void func(int type, char *data); // 宣告一個函式func p = func; // 將指標p指向函式func ......

    uj5u.com 2023-04-20 07:23:52 more
  • Django筆記二十五之資料庫函式之日期函式

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記二十五之資料庫函式之日期函式 日期函式主要介紹兩個大類,Extract() 和 Trunc() Extract() 函式作用是提取日期,比如我們可以提取一個日期欄位的年份,月份,日等資料 Trunc() 的作用則是截取,比如 2022-0 ......

    uj5u.com 2023-04-20 07:23:45 more
  • 一天吃透JVM面試八股文

    什么是JVM? JVM,全稱Java Virtual Machine(Java虛擬機),是通過在實際的計算機上仿真模擬各種計算機功能來實作的。由一套位元組碼指令集、一組暫存器、一個堆疊、一個垃圾回收堆和一個存盤方法域等組成。JVM屏蔽了與作業系統平臺相關的資訊,使得Java程式只需要生成在Java虛擬機 ......

    uj5u.com 2023-04-20 07:23:31 more
  • 使用Java接入小程式訂閱訊息!

    更新完微信服務號的模板訊息之后,我又趕緊把微信小程式的訂閱訊息給實作了!之前我一直以為微信小程式也是要企業才能申請,沒想到小程式個人就能申請。 訊息推送平臺🔥推送下發【郵件】【短信】【微信服務號】【微信小程式】【企業微信】【釘釘】等訊息型別。 https://gitee.com/zhongfuch ......

    uj5u.com 2023-04-20 07:22:59 more
  • java -- 緩沖流、轉換流、序列化流

    緩沖流 緩沖流, 也叫高效流, 按照資料型別分類: 位元組緩沖流:BufferedInputStream,BufferedOutputStream 字符緩沖流:BufferedReader,BufferedWriter 緩沖流的基本原理,是在創建流物件時,會創建一個內置的默認大小的緩沖區陣列,通過緩沖 ......

    uj5u.com 2023-04-20 07:22:49 more
  • Java-SpringBoot-Range請求頭設定實作視頻分段傳輸

    老實說,人太懶了,現在基本都不喜歡寫筆記了,但是網上有關Range請求頭的文章都太水了 下面是抄的一段StackOverflow的代碼...自己大修改過的,寫的注釋挺全的,應該直接看得懂,就不解釋了 寫的不好...只是希望能給視頻網站開發的新手一點點幫助吧. 業務場景:視頻分段傳輸、視頻多段傳輸(理 ......

    uj5u.com 2023-04-20 07:22:42 more
  • Windows 10開發教程_編程入門自學教程_菜鳥教程-免費教程分享

    教程簡介 Windows 10開發入門教程 - 從簡單的步驟了解Windows 10開發,從基本到高級概念,包括簡介,UWP,第一個應用程式,商店,XAML控制元件,資料系結,XAML性能,自適應設計,自適應UI,自適應代碼,檔案管理,SQLite資料庫,應用程式到應用程式通信,應用程式本地化,應用程式 ......

    uj5u.com 2023-04-20 07:22:35 more