全網最適合入門的面向對象編程教程:11 類和對象的Python實現-子類調用父類方法-模擬串口感測器和主機

来源:https://www.cnblogs.com/FreakEmbedded/p/18290960
-Advertisement-
Play Games

本節課,我們主要講解了在Python類的繼承中子類如何進行初始化、調用父類的屬性和方法,同時講解了模擬串口感測器和主機類的具體實現,並使用xcom串口助手與兩個類進行串口通信使用。 ...


全網最適合入門的面向對象編程教程:11 類和對象的 Python 實現-子類調用父類方法-模擬串口感測器和主機

摘要:

本節課,我們主要講解了在 Python 類的繼承中子類如何進行初始化、調用父類的屬性和方法,同時講解了模擬串口感測器和主機類的具體實現,並使用 xcom 串口助手與兩個類進行串口通信使用。

原文鏈接:

FreakStudio 的博客

往期推薦:

學嵌入式的你,還不會面向對象??!

全網最適合入門的面向對象編程教程:00 面向對象設計方法導論

全網最適合入門的面向對象編程教程:01 面向對象編程的基本概念

全網最適合入門的面向對象編程教程:02 類和對象的 Python 實現-使用 Python 創建類

全網最適合入門的面向對象編程教程:03 類和對象的 Python 實現-為自定義類添加屬性

全網最適合入門的面向對象編程教程:04 類和對象的Python實現-為自定義類添加方法

全網最適合入門的面向對象編程教程:05 類和對象的Python實現-PyCharm代碼標簽

全網最適合入門的面向對象編程教程:06 類和對象的Python實現-自定義類的數據封裝

全網最適合入門的面向對象編程教程:07 類和對象的Python實現-類型註解

全網最適合入門的面向對象編程教程:08 類和對象的Python實現-@property裝飾器

全網最適合入門的面向對象編程教程:09 類和對象的Python實現-類之間的關係

全網最適合入門的面向對象編程教程:10 類和對象的Python實現-類的繼承和里氏替換原則

更多精彩內容可看:

給你的 Python 加加速:一文速通 Python 並行計算

一文搞懂 CM3 單片機調試原理

肝了半個月,嵌入式技術棧大彙總出爐

電子電腦類比賽的“武林秘籍”

一個MicroPython的開源項目集錦:awesome-micropython,包含各個方面的Micropython工具庫

文檔和代碼獲取:

可訪問如下鏈接進行對文檔下載:

https://github.com/leezisheng/Doc

image

本文檔主要介紹如何使用 Python 進行面向對象編程,需要讀者對 Python 語法和單片機開發具有基本瞭解。相比其他講解 Python 面向對象編程的博客或書籍而言,本文檔更加詳細、側重於嵌入式上位機應用,以上位機和下位機的常見串口數據收發、數據處理、動態圖繪製等為應用實例,同時使用 Sourcetrail 代碼軟體對代碼進行可視化閱讀便於讀者理解。

相關示例代碼獲取鏈接如下:https://github.com/leezisheng/Python-OOP-Demo

正文

模擬串口感測器和主機類的具體實現:

接下來看一下我們新建的兩個類方法的具體實現,可以明確的是,SensorClass 和 MasterClass 都需要調用 SerialClass 類中有關串口收發的方法,也就是子類調用父類的方法,子類調用父類的方法有三種方式:

  • 父類名.方法名(self):此時需要加上父類的類名首碼,且需要帶上 self 參數變數。該方法單繼承或多繼承均適用。
  • super(子類名,self).父類方法名()/super().父類方法名:使用 super()函數,但如果涉及多繼承,該函數只能調用第一個直接父類的構造方法。

SensorClass 類的實現

(1)首先,我們將 SensorClass 中工作狀態的對應字元表示和命令對應字元表示設置為類屬性,什麼意思?我們來看如下代碼:

class SensorClass(SerialClass):
    _# 類變數:_
    _#   RESPOND_MODE-響應模式-0_
    _#   LOOP_MODE   -迴圈模式-1_
    RESPOND_MODE,LOOP_MODE = (0,1)
    _# 類變數:_
    _#   START_CMD       - 開啟命令      -0_
    _#   STOP_CMD        - 關閉命令      -1_
    _#   SENDID_CMD      - 發送ID命令    -2_
    _#   SENDVALUE_CMD   - 發送數據命令   -3_
    START_CMD,STOP_CMD,SENDID_CMD,SENDVALUE_CMD = (0,1,2,3)

類屬性歸類所有,與前面講的實例屬性不同,類屬性就相當與全局變數,是實例對象共有的屬性,類屬性影響類的所有對象;而實例對象的屬性為實例對象自己私有,實例屬性隻影響當前對象。類屬性常用於存儲常量、定義預設值或構造一個所有對象都能訪問的緩存。

這裡,我們定義了兩種類屬性:

RESPOND_MODE,LOOP_MODE = (0,1)

用於表示 SensorClass 不同工作模式。

START_CMD,STOP_CMD,SENDID_CMD,SENDVALUE_CMD = (0,1,2,3)

用於表示不同命令。

(2)在初始化中,我們調用父類初始化方法進行,同時可以在初始化 SensorClass 類時指定 id、state、port 三個參數。

def __init__(self,id:int = 0,state:int = RESPOND_MODE,port:str = "COM11"):
        # 調用父類的初始化方法,super() 函數將父類和子類連接
        super().__init__(port)
        self.sensorvalue = 0
        self.sensorid    = id
        self.sensorstate = state

這裡,實際上 SensorClass 類初始化的參數應該包括其他有關串口配置相關的參數(波特率、校驗位、數據位、停止位),由於串口通信雙方這些參數配置相同,這裡為了方便講解故而簡化。

(3)模擬感測器上電初始化,在實際感測器上電過程中會完成校準、自檢等操作,這裡我們簡單輸出感測器狀態和 ID 號:

_# 感測器上電初始化_
    def InitSensor(self):
        _# 感測器上電初始化工作_
        _# 同時輸出ID號以及狀態_
        print("Sensor %d Init complete : %d"%(self.sensorid,self.sensorstate))

(4)在感測器使能和關閉方法中,我們開啟或關閉串口並列印相關信息:

_# 開啟感測器_
    def StartSensor(self):
        super().OpenSerial()
        print("Sensor %d start serial %s "%(self.sensorid,self.dev.port))
        
    _# 停止感測器_
    def StopSensor(self):
        super().CloseSerial()
        print("Sensor %d close serial %s " % (self.sensorid, self.dev.port))

(5)在感測器發送 ID 號的方法中,我們調用了父類的 WriteSerial 方法:

_# 發送感測器ID號_
    def SendSensorID(self):
        super().WriteSerial(str(self.sensorid))
        print("Sensor %d send id "%self.sensorid)

(6)在感測器發送數據方法中,我們使用如下語句生成一個隨機數:

_# 生成[1, 10]內的隨機整數_
        data = random.randint(1, 10)

註意,此方法需要使用 import random 語句導入 random 庫。

同時調用父類的 WriteSerial 方法實現感測器數據的發送:

_# 發送感測器數據_
    def SendSensorValue(self):
        _# 生成[1, 10]內的隨機整數_
        data = random.randint(1, 10)
        super().WriteSerial(str(data))
        print("Sensor %d send data  %d" % (self.sensorid,data))

(7)在感測器接收命令方法中,我們調用了父類的 ReadSerial 接收命令:

_# 接收主機指令_
    def RecvMasterCMD(self):
        cmd = super().ReadSerial()
        print("Sensor %d recv cmd %d " % (self.sensorid,cmd))
        return cmd

MasterClass 類的實現

(1)首先定義關於工作模式和命令的類屬性:

class MasterClass(SerialClass):
    _# 類變數:_
    _#   BUSY_STATE  -忙碌狀態-0_
    _#   IDLE_STATE  -空閑狀態-1_
    BUSY_STATE, IDLE_STATE = (0, 1)
    _# 類變數:_
    _#   START_CMD       - 開啟命令      -0_
    _#   STOP_CMD        - 關閉命令      -1_
    _#   SENDID_CMD      - 發送ID命令    -2_
    _#   SENDVALUE_CMD   - 發送數據命令   -3_
    START_CMD, STOP_CMD, SENDID_CMD, SENDVALUE_CMD = (0, 1, 2, 3)

(2)在初始化函數中調用父類的初始化方法,並定義 valuequeue 和__masterstatue 屬性:

_# 類的初始化_
    def __init__(self,state:int = IDLE_STATE,port:str = "COM17"):
        _# 調用父類的初始化方法,super() 函數將父類和子類連接_
        super().__init__(port)
        self.valuequeue   = queue.Queue(10)
        self.__masterstatue = state

(3)在 StartMaster 方法中我們打開串口並使用 logging.info()方法輸出調試信息:

_# 開啟主機_
    def StartMaster(self):
        super().OpenSerial()
        logging.info("START MASTER :"+self.dev.port)

這裡,需要導入 logging 庫並設置日誌輸出級別:

import logging
_# 設置日誌輸出級別_
logging.basicConfig(level=logging.DEBUG)

(4)關閉主機方法中調用父類的 CloseSerial 方法:

_# 停止主機_
    def StopMaster(self):
        super().CloseSerial()
        logging.info("CLOSE MASTER :" + self.dev.port)

(5)如下調用父類的 ReadSerial 方法接收 ID 號和數據:

_# 接收感測器ID號_
    def RecvSensorID(self):
        sensorid = super().ReadSerial()
        logging.info("MASTER RECIEVE ID : " + str(sensorid))
        return sensorid

    _# 接收感測器數據_
    def RecvSensorValue(self):
        data = super().ReadSerial()
        logging.info("MASTER RECIEVE DATA : " + str(data))
        self.valuequeue.put(data)
        return data

(6)調用父類的 WriteSerial 方法發送指令:

_# 主機發送命令_
    def SendSensorCMD(self,cmd):
        super().WriteSerial(str(cmd))
        logging.info("MASTER SEND CMD : " + str(cmd))

(7)如下 RetMasterStatue 方法獲取主機狀態:

_# 主機返回工作狀態-_
    def RetMasterStatue(self):
        return self.__masterstatue

完整代碼

以下為兩個類的完整代碼:

class SensorClass(SerialClass):
    _# 類變數:_
    _#   RESPOND_MODE -響應模式-0_
    _#   LOOP_MODE    -迴圈模式-1_
    RESPOND_MODE,LOOP_MODE = (0,1)
    _# 類變數:_
    _#   START_CMD       - 開啟命令      -0_
    _#   STOP_CMD        - 關閉命令      -1_
    _#   SENDID_CMD      - 發送ID命令    -2_
    _#   SENDVALUE_CMD   - 發送數據命令   -3_
    START_CMD,STOP_CMD,SENDID_CMD,SENDVALUE_CMD = (0,1,2,3)

    _# 類的初始化_
    def __init__(self,port:str = "COM11",id:int = 0,state:int = RESPOND_MODE):
        _# 調用父類的初始化方法,super() 函數將父類和子類連接_
        super().__init__(port)
        self.sensorvalue = 0
        self.sensorid    = id
        self.sensorstate = state
    _# 感測器上電初始化_
    def InitSensor(self):
        _# 感測器上電初始化工作_
        _# 同時輸出ID號以及狀態_
        print("Sensor %d Init complete : %d"%(self.sensorid,self.sensorstate))
    _# 開啟感測器_
    def StartSensor(self):
        super().OpenSerial()
        print("Sensor %d start serial %s "%(self.sensorid,self.dev.port))
    _# 停止感測器_
    def StopSensor(self):
        super().CloseSerial()
        print("Sensor %d close serial %s " % (self.sensorid, self.dev.port))
    _# 發送感測器ID號_
    def SendSensorID(self):
        super().WriteSerial(str(self.sensorid))
        print("Sensor %d send id "%self.sensorid)
    _# 發送感測器數據_
    def SendSensorValue(self):
        _# 生成[1, 10]內的隨機整數_
        data = random.randint(1, 10)
        super().WriteSerial(str(data))
        print("Sensor %d send data  %d" % (self.sensorid,data))
    _# 接收主機指令_
    def RecvMasterCMD(self):
        cmd = super().ReadSerial()
        print("Sensor %d recv cmd %d " % (self.sensorid,cmd))
        return cmd

class MasterClass(SerialClass):
    _# 類變數:_
    _#   BUSY_STATE  -忙碌狀態-0_
    _#   IDLE_STATE  -空閑狀態-1_
    BUSY_STATE, IDLE_STATE = (0, 1)
    START_CMD, STOP_CMD, SENDID_CMD, SENDVALUE_CMD = (0, 1, 2, 3)

    _# 類的初始化_
    def __init__(self,state:int = IDLE_STATE,port:str = "COM17"):
        _# 調用父類的初始化方法,super() 函數將父類和子類連接_
        super().__init__(port)
        self.valuequeue   = queue.Queue(10)
        self.__masterstatue = state
    _# 開啟主機_
    def StartMaster(self):
        super().OpenSerial()
        logging.info("START MASTER :"+self.dev.port)
    _# 停止主機_
    def StopMaster(self):
        super().CloseSerial()
        logging.info("CLOSE MASTER :" + self.dev.port)
    _# 接收感測器ID號_
    def RecvSensorID(self):
        sensorid = super().ReadSerial()
        logging.info("MASTER RECIEVE ID : " + str(sensorid))
        return sensorid
    _# 接收感測器數據_
    def RecvSensorValue(self):
        data = super().ReadSerial()
        logging.info("MASTER RECIEVE DATA : " + str(data))
        self.valuequeue.put(data)
        return data
    _# 主機發送命令_
    def SendSensorCMD(self,cmd):
        super().WriteSerial(str(cmd))
        logging.info("MASTER SEND CMD : " + str(cmd))
    _# 主機返回工作狀態-_
    def RetMasterStatue(self):
        return self.__masterstatue

模擬實例

這裡,我們使用 XCOM 軟體和我們的 Python 程式進行交互。

感測器實驗模擬

這裡,我們首先在主函數中創建感測器對象,完成初始化後使能感測器中串口模塊,並設置迴圈,輪詢讀取指令並執行操作,示例代碼如下:

if __name__ == "__main__":
    _# 創建感測器對象_
    s = SensorClass(port="COM11", id=1, state=SensorClass.RESPOND_MODE)
    _# 初始化感測器_
    s.InitSensor()
    _# 感測器開啟_
    s.StartSensor()

    while True:
        _# 根據不同指令執行不同操作_
        cmd = s.RecvMasterCMD()
        _# START_CMD, STOP_CMD, SENDID_CMD, SENDVALUE_CMD = (0, 1, 2, 3)_
        if cmd == SensorClass.SENDID_CMD:
            s.SendSensorID()
        elif cmd == SensorClass.SENDVALUE_CMD:
            s.SendSensorValue()
        elif cmd == SensorClass.STOP_CMD:
            s.StopSensor()
            break
    print(" Sensor Stop Work!")

我們來看一下實際驗證效果:

image

主機實驗模擬

這裡,我們首先在主函數中創建並開啟主機對象,我們的 xcom 模擬感測器,主機在輪詢中發送接收數據指令,並將接收的數據加入主機類的隊列,最後發送停機命令,並關閉主機。

if __name__ == "__main__":
    m = MasterClass(state=MasterClass.IDLE_STATE, port="COM17")
    m.StartMaster()

    _# START_CMD, STOP_CMD, SENDID_CMD, SENDVALUE_CMD = (0, 1, 2, 3)_
    _# 發送指令,獲取感測器ID_
    m.SendSensorCMD(MasterClass.SENDID_CMD)
    m.RecvSensorID()

    for i in range(3):
        _# 發送指令,獲取感測器數據_
        m.SendSensorCMD(MasterClass.SENDVALUE_CMD)
        m.RecvSensorValue()

    m.SendSensorCMD(MasterClass.STOP_CMD)
    m.StopMaster()

    print("Master Stop Work!")

這裡我們將主機日誌列印到文件中:

_# 在配置下日誌輸出目標文件和日誌格式_
LOG_FORMAT = "%(asctime)s - %(levelname)s - %(message)s"
logging.basicConfig(filename='my.log', level=logging.DEBUG, format=LOG_FORMAT)

我們來看一下實際驗證效果:

image

可以看到兩個實驗都可以完整運行,關於兩個類的交互工作,我們將在後續多線程中講到。

image


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

-Advertisement-
Play Games
更多相關文章
  • title: Nuxt框架中內置組件詳解及使用指南(一) date: 2024/7/6 updated: 2024/7/6 author: cmdragon excerpt: 本文詳細介紹了Nuxt框架中的兩個內置組件和的使用方法與功能。確保包裹的內容僅在客戶端渲染,適用於處理瀏覽器特定功能或非同步數 ...
  • 本文探討了JavaScript中Promise的基礎用法和各種靜態方法的應用場景。從解決非同步編程中的回調地獄問題,到鏈式調用、併發請求控制,再到最新的Promise.allSettled和Promise.any的應用。每種方法均通過代碼示例和詳細的應用場景進行了展示。 ...
  • 摘要:本文詳細介紹了Nuxt3中幾個關鍵的生命周期鉤子和它們的使用方法,包括webpack:done用於Webpack編譯完成後執行操作,webpack:progress監聽編譯進度,render:response和render:html分別在響應發送前後修改響應內容,以及render:island... ...
  • ‍ 寫在開頭 點贊 + 收藏 學會 7 種方案解決移動端1px邊框的問題 造成邊框變粗的原因 css中的1px並不等於移動設備的1px,這是由不同手機由不同像素密度,在window對象中有一個devicePixelRatio屬性,他可以反應css中像素與設備的像素比 device ...
  • ## 動態表單系統:利用 Spring Boot 和 MyBatis 實現後端服務 在現代企業應用中,表單是數據收集和處理的核心部分。然而,傳統的表單系統難以適應快速變化的需求。為瞭解決這個問題,我們可以使用動態表單系統,它可以根據業務需求靈活地調整表單結構。本文將介紹如何使用 Spring Boo ...
  • 本文探討了Node.js事件迴圈中的timers階段,分析了定時器的管理和執行過程。通過源碼解析,揭示了定時器超時檢查、回調執行以及定時器類型(setTimeout與setInterval)的內部實現機制。文章旨在幫助讀者理解Node.js中定時器的工作原理及其在事件驅動編程中的重要性。 ...
  • 摘要:本文詳細介紹了Nuxt3框架中的五個webpack鉤子函數:webpack:configResolved用於在webpack配置解析後讀取和修改配置;webpack:compile在編譯開始前調用,可修改編譯選項;webpack:compiled在編譯完成後調用,可處理編譯結果;webpack... ...
  • ‍ 寫在開頭 點贊 + 收藏 學會 前言: 最近接到了一個需求很有意思,類似於我們經常在逛購物平臺中,選擇一個物品分享給好友,然後好友複製這段文本打開相對應的平臺以後,就可以彈出鏈接上的物品。實現過程也比較有意思,特來分享一下實現思路。 一. 效果預覽 當我在別的界面複製了內 ...
一周排行
    -Advertisement-
    Play Games
  • 前言 微服務架構已經成為搭建高效、可擴展系統的關鍵技術之一,然而,現有許多微服務框架往往過於複雜,使得我們普通開發者難以快速上手並體驗到微服務帶了的便利。為瞭解決這一問題,於是作者精心打造了一款最接地氣的 .NET 微服務框架,幫助我們輕鬆構建和管理微服務應用。 本框架不僅支持 Consul 服務註 ...
  • 先看一下效果吧: 如果不會寫動畫或者懶得寫動畫,就直接交給Blend來做吧; 其實Blend操作起來很簡單,有點類似於在操作PS,我們只需要設置關鍵幀,滑鼠點來點去就可以了,Blend會自動幫我們生成我們想要的動畫效果. 第一步:要創建一個空的WPF項目 第二步:右鍵我們的項目,在最下方有一個,在B ...
  • Prism:框架介紹與安裝 什麼是Prism? Prism是一個用於在 WPF、Xamarin Form、Uno 平臺和 WinUI 中構建鬆散耦合、可維護和可測試的 XAML 應用程式框架 Github https://github.com/PrismLibrary/Prism NuGet htt ...
  • 在WPF中,屏幕上的所有內容,都是通過畫筆(Brush)畫上去的。如按鈕的背景色,邊框,文本框的前景和形狀填充。藉助畫筆,可以繪製頁面上的所有UI對象。不同畫筆具有不同類型的輸出( 如:某些畫筆使用純色繪製區域,其他畫筆使用漸變、圖案、圖像或繪圖)。 ...
  • 前言 嗨,大家好!推薦一個基於 .NET 8 的高併發微服務電商系統,涵蓋了商品、訂單、會員、服務、財務等50多種實用功能。 項目不僅使用了 .NET 8 的最新特性,還集成了AutoFac、DotLiquid、HangFire、Nlog、Jwt、LayUIAdmin、SqlSugar、MySQL、 ...
  • 本文主要介紹攝像頭(相機)如何採集數據,用於類似攝像頭本地顯示軟體,以及流媒體數據傳輸場景如傳屏、視訊會議等。 攝像頭採集有多種方案,如AForge.NET、WPFMediaKit、OpenCvSharp、EmguCv、DirectShow.NET、MediaCaptre(UWP),網上一些文章以及 ...
  • 前言 Seal-Report 是一款.NET 開源報表工具,擁有 1.4K Star。它提供了一個完整的框架,使用 C# 編寫,最新的版本採用的是 .NET 8.0 。 它能夠高效地從各種資料庫或 NoSQL 數據源生成日常報表,並支持執行複雜的報表任務。 其簡單易用的安裝過程和直觀的設計界面,我們 ...
  • 背景需求: 系統需要對接到XXX官方的API,但因此官方對接以及管理都十分嚴格。而本人部門的系統中包含諸多子系統,系統間為了穩定,程式間多數固定Token+特殊驗證進行調用,且後期還要提供給其他兄弟部門系統共同調用。 原則上:每套系統都必須單獨接入到官方,但官方的接入複雜,還要官方指定機構認證的證書 ...
  • 本文介紹下電腦設備關機的情況下如何通過網路喚醒設備,之前電源S狀態 電腦Power電源狀態- 唐宋元明清2188 - 博客園 (cnblogs.com) 有介紹過遠程喚醒設備,後面這倆天瞭解多了點所以單獨加個隨筆 設備關機的情況下,使用網路喚醒的前提條件: 1. 被喚醒設備需要支持這WakeOnL ...
  • 前言 大家好,推薦一個.NET 8.0 為核心,結合前端 Vue 框架,實現了前後端完全分離的設計理念。它不僅提供了強大的基礎功能支持,如許可權管理、代碼生成器等,還通過採用主流技術和最佳實踐,顯著降低了開發難度,加快了項目交付速度。 如果你需要一個高效的開發解決方案,本框架能幫助大家輕鬆應對挑戰,實 ...