串口通訊學習

来源:https://www.cnblogs.com/namelessblog/archive/2020/07/01/13221303.html
-Advertisement-
Play Games

//前提需要 //需要一個 serialPort 工具 可在vs自帶的工具欄中獲得 //源代碼加串口工具地址: //鏈接:https://pan.baidu.com/s/1YbfvdXEmfsJX87D-Jxljyg 提取碼:d32x //記錄用戶打開的串口號 可改為泛型模式 string seri ...


//前提需要

//需要一個 serialPort 工具 可在vs自帶的工具欄中獲得

//源代碼加串口工具地址:

//鏈接:https://pan.baidu.com/s/1YbfvdXEmfsJX87D-Jxljyg   提取碼:d32x

 

//記錄用戶打開的串口號 可改為泛型模式

string serialPortName;

//一般電腦是沒有串口號的,所以要用個軟體創建虛擬串口號

//軟體名 - Configure Virtual Serial Port Driver

 

波特率設置的值為 1382400,921600,460800,256000,230400,128000,115200,76800,57600,

43000,38400,19200,14400,9600,4800,1200

停止位設置的值為:1,1.5,2

數據位設置的值為:8,7,6,5

校驗位設置的值為:無,奇校驗,偶校驗

 

步驟:

1.設置串口的基本數據 (串口號,波特率,數據位,停止位,校驗位)

2.使用Open打開串口

3.使用串口工具的DataReceived()事件進行數據的接收

4.創建發送數據 Write() ,一般發送的是byte數組

5.使用Close關閉串口

 

界面:

 

 

 

//一

//在視窗Load事件中,設置串口基本數據的預設值

//1.拿到電腦上可以使用的串口號,並賦值給視窗的串口下拉列表中,預設選項索引為0

string[] prots = System.IO.Ports.SerialPort.GetPortNames();

this.cbPort.Items.AddRange(prots);

this.cbPort.SelectedIndex = this.cbPort.Items.Count > 0 ? 0 : -1;

//2.設置波特率,停止位,數據位,校驗位的預設值。

this.cbBaud.Text = "115200";

this.cbStopBit.Text = "1";

this.cbDataBit.Text = "8";

this.cbComparable.Text = "無";

 

 

//二

//設置打開串口和關閉串口按鈕的單擊事件

//為串口拿到基本數據

serialPort .PortName = this.cbPort.Text;//串口號

serialPortName = this.cbPort.Text;//記錄用戶打開的串口號

 

serialPort.BaudRate = int.Parse(this.cbBaud.Text);//波特率

serialPort.DataBits = int.Parse(this.cbDataBit.Text);//數據位

//設置停止位

if (this.cbStopBit.Text == "1") { serialPort.StopBits = StopBits.One; }

else if (this.cbStopBit.Text == "1.5") { serialPort.StopBits = StopBits.OnePointFive; }

else if (this.cbStopBit.Text == "2") { serialPort.StopBits = StopBits.Two; }

//設置奇偶校驗

if (this.cbComparable.Text == "無") { serialPort.Parity = Parity.None; }

else if (this.cbComparable.Text == "奇校驗") { serialPort.Parity = Parity.Odd; }

else if (this.cbComparable.Text == "偶校驗") { serialPort.Parity = Parity.Even; }

 

//打開串口 只需要使用Open打開串口

serialPort1.Open();

 

this.btnOpenStat.Text = "關閉串口";

 

//如果按鈕的Text為 關閉串口的話

serialPort.Close();//Close() - 關閉串口

 

 

//三 本次串口的簡單數據協議 1 - 字元串 bp - 醫療數據 2 - 文件

//通過串口工具自帶的DataReceived事件進行串口的數據接收

//這次的是自己定義的數據協議,一般都是用公司預設的協議 或者RS232和RS485

//獲取可以讀取的位元組數 預設最大值是4096 如果想要更改可以直接改變串口的ReadBufferSize屬性

int len = serialPort.BytesToRead;

if(len == serialPort.ReadBufferSize){

MessageBox.Show("程式最多只能接受"+serialPort1.ReadBufferSize+"位元組的數據!","提示");

return;

}

//拿到數據,Read()

byte[] buff = new byte[len];

serialPort.Read(buff,0,len);

//如果選中了顯示時間就賦值,沒有就算

string DateNow = string.Empty;

//根據協議進行判斷

if(buff[0]==1){

//拿出原來的數據並去除協議部分 然後放入另一個byte[]數組中

byte[] result = new byte[len];

Buffer.BlockCopy(buff,1,result,0,buff.Length-1);

//將byte數組保存可讀string 根據自己的編碼格式進行轉碼

string str = Encoding.Default.GetString(result);

//將數據分割為string[] 判斷是否為醫療數據

string[] strPle = str.Split(',');

if(strPle[0]=="bp"){

//放入方法中,重新拼裝成可用數據

str = RecHBpData(strPle);

}

//由於我們的接收數據的事件是在一個子線程裡面的,

//所以需要Invoke才能給我們主線程創建的控制項賦值

//當然你也可以使用InvokeRequired()來判斷是否為主線程創建的控制項,我這裡就沒使用了

Invoke(new Action(() => {

//是否顯示時間

if(this.ckDateTime.Checked){

DateNow = "\r\n" + DateTime.Now.ToString();

}

//是否為16進位顯示

if(this.checkBox1.Checked){

this.txtMsg.AppendText(DateNow + "\r\n" + byteToHexStr(buff));

}

//預設是使用字元串顯示

else{

this.txtMsg.AppendText(DateNow + "\r\n" + str);

}

 

})));

}

//如果傳送的文件的話

else if(buff[0]==2){

ProcessRecieveFile(buff);

}

 

//一般呢,這些方法是放在一個方法類庫裡面,但是我懶得放就在一個類裡面了

#region 封裝的方法

 

/// <summary>

/// 傳輸的是文件就進行文件的操作

/// </summary>

/// <param name="data">數據流</param>

public void ProcessRecieveFile(byte[] data)

{

using (SaveFileDialog dialog = new SaveFileDialog())

{

dialog.DefaultExt = "txt";

dialog.Filter = "文本文件(*.txt)|*.txt|所有文件(*.*)|*.*";

//由於大部分可能不是主線程,所以我們要加this,不然不會彈出保存框

if (dialog.ShowDialog(this) != DialogResult.OK)

{

return;

}

byte[] result = new byte[data.Length - 1];

Buffer.BlockCopy(data, 1, result, 0, data.Length - 1);

File.WriteAllBytes(dialog.FileName, result);

}

}

 

#region 封裝 數據定義的協議

 

public static string RecHBpData(string[] strPle)

{

string str = string.Empty;

 

//判斷儀器

str += $"{StaticConstant.DicMedicalnoun[strPle[1]]} ";

 

//增加時間

str += " " + strPle[2] + " " + strPle[3] + "\r\n";

//增加具體數據

str += $"{StaticConstant.DicMedicalnoun[strPle[4]]}: {strPle[5].TrimStart('0')} \r\n{StaticConstant.DicMedicalnoun[strPle[6]]}: {strPle[7].TrimStart('0')}";

 

 

 

 

 

return str;

}

 

 

#endregion

 

 

/// <summary>

/// 位元組數組轉16進位字元串

/// </summary>

/// <param name="bytes">byte數組</param>

/// <returns>16進位顯示形式</returns>

public static string byteToHexStr(byte[] bytes)

{

string retuenStr = "";

try

{

if (bytes != null)

{

for (int i = 0; i < bytes.Length; i++)

{

retuenStr += bytes[i].ToString("X2") + " ";//變成16進位,兩個中間用空格隔開

}

}

return retuenStr;

}

catch (Exception ex)

{

return retuenStr;

}

}

 

/// <summary>

/// 字元串轉16進位格式,不夠自動前面補0

/// </summary>

/// <param name="hexString"></param>

/// <returns></returns>

public static byte[] strToHexByte(string hexString)

{

int i;

if ((hexString.Length % 2) != 0)

{//奇數個

byte[] returnBytes = new byte[(hexString.Length + 1) / 2];

try

{

for (i = 0; i < hexString.Length - 1; i++)

{

returnBytes[i] = Convert.ToByte(hexString.Substring(i * 2, 2), 16);

i = i * 2;

}

returnBytes[returnBytes.Length - 1] = Convert.ToByte(hexString.Substring(hexString.Length - 1, 1).PadLeft(2, '0'), 16);

}

catch (Exception)

{

MessageBox.Show("含有非16進位字元", "提示");

return null;

}

return returnBytes;

}

else

{

byte[] returnBytes = new byte[(hexString.Length) / 2];

try

{

for (i = 0; i < hexString.Length; i++)

{

returnBytes[i] = Convert.ToByte(hexString.Substring(i * 2, 2), 16);

}

}

catch (Exception)

{

MessageBox.Show("含有非16進位字元", "提示");

return null;

}

return returnBytes;

}

}

 

#endregion

 

 

//我的醫療數據的封裝是放在配置文件中,一般我們都是放在資料庫中,不過我就做個小學習工具就 //沒做到資料庫裡面了

 

//四 發送數據

//發送數據,我這邊分為發送文件和發送字元串

//1.發送字元串

string str = this.txtSendMsg.Text.Trim().Tostring();

try{

if(str.Length > 0){

//16進位發送

if(this.ckSend.Checked){

byte[] byt = strToHexBytes(str);

byte[] result = new byte[byt.Length+1];

//加上數據協議

result[0] = 1;

//數據轉移

Buffer.BlockCopy(byt,0,result,1,byt.Lengt);

//發送數據,byte形式發送

serialPort.Write(result,0,result.Length);

}

//預設字元串發送

else{

//根據自己的編碼格式編碼

byte[] dataByte = Encoding.Default.GetBytes(str);

byte[] result = new byte[dataByte.Length + 1];

//加上數據協議

result[0] = 1;

Buffer.BlockCopy(dataByte ,0,result,1,dataByte .Lengt);

//發送數據,byte形式發送

serialPort.Write(result,0,result.Length);

}

}

}catch(Exception ex){}

 

//2.發送文件

try{

using(OpenFileDialog dialog = new OpenFileDialog()){

if(dialog.ShowDialog(this) != DialogResult.OK){

return;

}

byte[] data = File.ReadAllBytes(dialog.FileName);

byte[] result = new byte[data.Length + 1];

result[0] = 2;

Buffer.BlockCopy(data,0,result,1,data.Length);

serialPort.Write(result,0,result.Lenght);

}

}catch(Exception ex){

MessageBox.Show(ex.Message);

}

 


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

-Advertisement-
Play Games
更多相關文章
  • 一.通過console.log輸出(我最喜歡的) 1.js腳本 1.js var arguments = process.argv.splice(2); //獲得入參 var a= arguments[0]; 取第一個 console.log(a) //輸出 2.python腳本 test_1.py ...
  • numpy廣播(Broadcast) 若數組a,b形狀相同,即a.shape==b.shape,那麼a+b,a*b的結果就是對應數位的運算 import numpy as np a=np.array([[1,2,3],[4,5,6]]) b=np.array([[1,4,7],[2,5,8]]) p ...
  • 不知不覺中,做Java開發已經四年了,趁著現在換了工作,還算比較閑,就整理一下自己學習的一些知識。 在學習過程中,我喜歡找的一些電子書來系統化的學習,多數時候我都只是做一個大概的瞭解,沒有深入學習,基本就是看下目錄,看下前幾章的內容,深入的只會在實際用的時候才去詳細看了。我從以下幾個方面整理了一下這 ...
  • 一.node啟動js-3DES-ECB加密 var arguments = process.argv.splice(2); // console.log('所傳遞的參數是:', arguments); var password = arguments[0]; //加密的password var t= ...
  • LeetCode–最長公共首碼 博客說明 文章所涉及的資料來自互聯網整理和個人總結,意在於個人學習和經驗彙總,如有什麼地方侵權,請聯繫本人刪除,謝謝! 說明 leetcode題,14題 最長公共首碼 題目 編寫一個函數來查找字元串數組中的最長公共首碼。 如果不存在公共首碼,返回空字元串 ""。 示例 ...
  • 一.node啟動js公鑰加密 //需要導入模塊npm install node-forge var arguments = process.argv.splice(2); // console.log('所傳遞的參數是:', arguments); var e = arguments[1]; var ...
  • Java電子書分類 Java、Spring、SpringBoot、SpringCloud、mybatis、Tomcat、多線程、Git相關、Redis、設計模式、Nginx、Linux、演算法、資料庫、大數據、架構 電子書大概有30G左右 部分電子書截圖 領取方式 加我的微信(s2001sssss)免 ...
  • 從業務視角還原問題、業務視角抽象問題、二次抽象發現技術問題,作為思考問題的三個還原點。 提煉單點問題解決能力、思考單點問題解決能力,複製、層層遞進思考問題,從深度和廣度出發, 深度以數據作指標,廣度以領域做方向。 腳踏實地的代碼量成長沒有捷徑。 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...