在Python中,協議(Protocol)和介面(Interface)是用於定義類和對象之間交互的一種方式,特別是在實現多態性和代碼可重用性時,協議是一種抽象概念,描述了對象所需實現的方法和屬性,而不關心具體的類或實現。 ...
全網最適合入門的面向對象編程教程:52 Python 函數方法與介面-Protocol 協議與介面
摘要:
在 Python 中,協議(Protocol)和介面(Interface)是用於定義類和對象之間交互的一種方式,特別是在實現多態性和代碼可重用性時,協議是一種抽象概念,描述了對象所需實現的方法和屬性,而不關心具體的類或實現。
原文鏈接:
往期推薦:
全網最適合入門的面向對象編程教程:00 面向對象設計方法導論
全網最適合入門的面向對象編程教程:01 面向對象編程的基本概念
全網最適合入門的面向對象編程教程:02 類和對象的 Python 實現-使用 Python 創建類
全網最適合入門的面向對象編程教程:03 類和對象的 Python 實現-為自定義類添加屬性
全網最適合入門的面向對象編程教程:04 類和對象的Python實現-為自定義類添加方法
全網最適合入門的面向對象編程教程:05 類和對象的Python實現-PyCharm代碼標簽
全網最適合入門的面向對象編程教程:06 類和對象的Python實現-自定義類的數據封裝
全網最適合入門的面向對象編程教程:07 類和對象的Python實現-類型註解
全網最適合入門的面向對象編程教程:08 類和對象的Python實現-@property裝飾器
全網最適合入門的面向對象編程教程:09 類和對象的Python實現-類之間的關係
全網最適合入門的面向對象編程教程:10 類和對象的Python實現-類的繼承和里氏替換原則
全網最適合入門的面向對象編程教程:11 類和對象的Python實現-子類調用父類方法
全網最適合入門的面向對象編程教程:12 類和對象的Python實現-Python使用logging模塊輸出程式運行日誌
全網最適合入門的面向對象編程教程:13 類和對象的Python實現-可視化閱讀代碼神器Sourcetrail的安裝使用
全網最適合入門的面向對象編程教程:全網最適合入門的面向對象編程教程:14 類和對象的Python實現-類的靜態方法和類方法
全網最適合入門的面向對象編程教程:15 類和對象的 Python 實現-__slots__魔法方法
全網最適合入門的面向對象編程教程:16 類和對象的Python實現-多態、方法重寫與開閉原則
全網最適合入門的面向對象編程教程:17 類和對象的Python實現-鴨子類型與“file-like object“
全網最適合入門的面向對象編程教程:18 類和對象的Python實現-多重繼承與PyQtGraph串口數據繪製曲線圖
全網最適合入門的面向對象編程教程:19 類和對象的 Python 實現-使用 PyCharm 自動生成文件註釋和函數註釋
全網最適合入門的面向對象編程教程:20 類和對象的Python實現-組合關係的實現與CSV文件保存
全網最適合入門的面向對象編程教程:21 類和對象的Python實現-多文件的組織:模塊module和包package
全網最適合入門的面向對象編程教程:22 類和對象的Python實現-異常和語法錯誤
全網最適合入門的面向對象編程教程:23 類和對象的Python實現-拋出異常
全網最適合入門的面向對象編程教程:24 類和對象的Python實現-異常的捕獲與處理
全網最適合入門的面向對象編程教程:25 類和對象的Python實現-Python判斷輸入數據類型
全網最適合入門的面向對象編程教程:26 類和對象的Python實現-上下文管理器和with語句
全網最適合入門的面向對象編程教程:27 類和對象的Python實現-Python中異常層級與自定義異常類的實現
全網最適合入門的面向對象編程教程:28 類和對象的Python實現-Python編程原則、哲學和規範大彙總
全網最適合入門的面向對象編程教程:29 類和對象的Python實現-斷言與防禦性編程和help函數的使用
全網最適合入門的面向對象編程教程:30 Python的內置數據類型-object根類
全網最適合入門的面向對象編程教程:31 Python的內置數據類型-對象Object和類型Type
全網最適合入門的面向對象編程教程:32 Python的內置數據類型-類Class和實例Instance
全網最適合入門的面向對象編程教程:33 Python的內置數據類型-對象Object和類型Type的關係
全網最適合入門的面向對象編程教程:34 Python的內置數據類型-Python常用複合數據類型:元組和命名元組
全網最適合入門的面向對象編程教程:35 Python的內置數據類型-文檔字元串和__doc__屬性
全網最適合入門的面向對象編程教程:36 Python的內置數據類型-字典
全網最適合入門的面向對象編程教程:37 Python常用複合數據類型-列表和列表推導式
全網最適合入門的面向對象編程教程:38 Python常用複合數據類型-使用列表實現堆棧、隊列和雙端隊列
全網最適合入門的面向對象編程教程:39 Python常用複合數據類型-集合
全網最適合入門的面向對象編程教程:40 Python常用複合數據類型-枚舉和enum模塊的使用
全網最適合入門的面向對象編程教程:41 Python常用複合數據類型-隊列(FIFO、LIFO、優先順序隊列、雙端隊列和環形隊列)
全網最適合入門的面向對象編程教程:42 Python常用複合數據類型-collections容器數據類型
全網最適合入門的面向對象編程教程:43 Python常用複合數據類型-擴展內置數據類型
全網最適合入門的面向對象編程教程:44 Python內置函數與魔法方法-重寫內置類型的魔法方法
全網最適合入門的面向對象編程教程:45 Python實現常見數據結構-鏈表、樹、哈希表、圖和堆
全網最適合入門的面向對象編程教程:46 Python函數方法與介面-函數與事件驅動框架
全網最適合入門的面向對象編程教程:47 Python函數方法與介面-回調函數Callback
全網最適合入門的面向對象編程教程:48 Python函數方法與介面-位置參數、預設參數、可變參數和關鍵字參數
全網最適合入門的面向對象編程教程:49 Python函數方法與介面-函數與方法的區別和lamda匿名函數
全網最適合入門的面向對象編程教程:50 Python函數方法與介面-介面和抽象基類
全網最適合入門的面向對象編程教程:51 Python函數方法與介面-使用Zope實現介面
更多精彩內容可看:
給你的 Python 加加速:一文速通 Python 並行計算
一個MicroPython的開源項目集錦:awesome-micropython,包含各個方面的Micropython工具庫
SenseCraft 部署模型到Grove Vision AI V2圖像處理模塊
文檔和代碼獲取:
可訪問如下鏈接進行對文檔下載:
https://github.com/leezisheng/Doc
本文檔主要介紹如何使用 Python 進行面向對象編程,需要讀者對 Python 語法和單片機開發具有基本瞭解。相比其他講解 Python 面向對象編程的博客或書籍而言,本文檔更加詳細、側重於嵌入式上位機應用,以上位機和下位機的常見串口數據收發、數據處理、動態圖繪製等為應用實例,同時使用 Sourcetrail 代碼軟體對代碼進行可視化閱讀便於讀者理解。
相關示例代碼獲取鏈接如下:https://github.com/leezisheng/Python-OOP-Demo
正文
協議 protocol 在 python3.8 開始引入的,協議是抽象基類 (ABC) 的替代方案,作為在語言中定義結構類型或“鴨子類型”的一種方式。協議是對象必須具有的一組方法或屬性,以便被認為與該協議相容。協議使你能夠定義介面,而無需顯式創建類或從特定基類繼承。
協議是使用 typing.Protocol 類或 typing.Protocol 裝飾器定義的。
以下為示例代碼,首先我們定義一個名為 Printable 的協議,該協議要求對象具有列印方法。 Protocol 類允許我們定義必須由符合協議的對象實現的抽象方法(在本例中為列印)。
from typing import Protocol
class Printable(Protocol):
def print(self) -> None:
pass
要實現協議,我們不需要顯式聲明它。相反,我們可以通過實現所需的方法來確保對象符合協議。這是一個例子:在此示例中,Book 類通過提供所需的列印方法來實現 Printable 協議。Book 類的任何實例都可以被視為 Printable 對象。
協議主要用於類型提示和靜態類型檢查。我們可以使用協議作為類型註釋來指示參數或變數必須符合特定協議。這是一個例子:
def print_object(obj: Printable) -> None:
obj.print()
在此示例中,print_object 函數採用 Printable 類型的參數 obj。這意味著傳遞給此函數的任何對象都必須實現 Printable 協議中定義的 print 方法。
接下來,我們定義一個沒有 print 方法的類
_# 定義CD類,沒有列印方法_
class CD:
def __init__(self, title: str):
self.title = title
def play(self) -> None:
print(f"play music Title: {self.title}")
接下來,實例化兩個類,並且測試是否能使用 print_object 函數:
book_obj = Book("Piece and Love")
print_object(book_obj)
cd_obj = CD("Piece and Love")
print_object(cd_obj)
運行結果如下:
在 IDE 中實際上也會有相關提示:
實際上,協議的實質含義可以理解為“這個類只要有 xxxx 方法,那麼它就會隱式地成為我的子類”,這個東西主要是給靜態類型檢查工具看的,性質上和 PEP 484 與 PEP 526 是一樣的。我們也可以利用 @typing.runtime_checkable 裝飾器將協議類標記為運行時協議,這種協議可以與 isinstance()和 issubclass()一起使用。當應用於非協議類時,這會引發 TypeError。
_# 導入協議Protocol_
from typing import Protocol
import typing
_# 定義Printable協議,需要具有列印方法_
@typing.runtime_checkable
class Printable(Protocol):
def print(self) -> None:
pass
... ...
_# 創建對象實例_
book_obj = Book("Piece and Love")
print_object(book_obj)
cd_obj = CD("Piece and Love")
_# 類型檢查_
print(isinstance(Book,Printable))
assert isinstance(book_obj,Printable)
print(isinstance(CD,Printable))
assert isinstance(cd_obj,Printable)
運行結果如下:
我們也可以定義組合多個其他協議的協議。以下我們使用序列化方法定義了一個名為 Serialized 的協議。然後我們定義另一個名為 PrintableAndSerializing 的協議,它結合了 Printable 和 Serializing 協議。
class Serializable(Protocol):
def serialize(self) -> str:
pass
class PrintableAndSerializable(Printable, Serializable):
pass
相比抽象基類等方式,協議提供了一種在 Python 中定義結構類型的方法,允許我們創建介面而無需顯式繼承。它們使類型檢查器能夠動態驗證對象是否遵守定義的協議,從而增強代碼的正確性和可維護性。