串口通訊學習

来源: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
  • Timer是什麼 Timer 是一種用於創建定期粒度行為的機制。 與標準的 .NET System.Threading.Timer 類相似,Orleans 的 Timer 允許在一段時間後執行特定的操作,或者在特定的時間間隔內重覆執行操作。 它在分散式系統中具有重要作用,特別是在處理需要周期性執行的 ...
  • 前言 相信很多做WPF開發的小伙伴都遇到過表格類的需求,雖然現有的Grid控制項也能實現,但是使用起來的體驗感並不好,比如要實現一個Excel中的表格效果,估計你能想到的第一個方法就是套Border控制項,用這種方法你需要控制每個Border的邊框,並且在一堆Bordr中找到Grid.Row,Grid. ...
  • .NET C#程式啟動閃退,目錄導致的問題 這是第2次踩這個坑了,很小的編程細節,容易忽略,所以寫個博客,分享給大家。 1.第一次坑:是windows 系統把程式運行成服務,找不到配置文件,原因是以服務運行它的工作目錄是在C:\Windows\System32 2.本次坑:WPF桌面程式通過註冊表設 ...
  • 在分散式系統中,數據的持久化是至關重要的一環。 Orleans 7 引入了強大的持久化功能,使得在分散式環境下管理數據變得更加輕鬆和可靠。 本文將介紹什麼是 Orleans 7 的持久化,如何設置它以及相應的代碼示例。 什麼是 Orleans 7 的持久化? Orleans 7 的持久化是指將 Or ...
  • 前言 .NET Feature Management 是一個用於管理應用程式功能的庫,它可以幫助開發人員在應用程式中輕鬆地添加、移除和管理功能。使用 Feature Management,開發人員可以根據不同用戶、環境或其他條件來動態地控制應用程式中的功能。這使得開發人員可以更靈活地管理應用程式的功 ...
  • 在 WPF 應用程式中,拖放操作是實現用戶交互的重要組成部分。通過拖放操作,用戶可以輕鬆地將數據從一個位置移動到另一個位置,或者將控制項從一個容器移動到另一個容器。然而,WPF 中預設的拖放操作可能並不是那麼好用。為瞭解決這個問題,我們可以自定義一個 Panel 來實現更簡單的拖拽操作。 自定義 Pa ...
  • 在實際使用中,由於涉及到不同編程語言之間互相調用,導致C++ 中的OpenCV與C#中的OpenCvSharp 圖像數據在不同編程語言之間難以有效傳遞。在本文中我們將結合OpenCvSharp源碼實現原理,探究兩種數據之間的通信方式。 ...
  • 一、前言 這是一篇搭建許可權管理系統的系列文章。 隨著網路的發展,信息安全對應任何企業來說都越發的重要,而本系列文章將和大家一起一步一步搭建一個全新的許可權管理系統。 說明:由於搭建一個全新的項目過於繁瑣,所有作者將挑選核心代碼和核心思路進行分享。 二、技術選擇 三、開始設計 1、自主搭建vue前端和. ...
  • Csharper中的表達式樹 這節課來瞭解一下表示式樹是什麼? 在C#中,表達式樹是一種數據結構,它可以表示一些代碼塊,如Lambda表達式或查詢表達式。表達式樹使你能夠查看和操作數據,就像你可以查看和操作代碼一樣。它們通常用於創建動態查詢和解析表達式。 一、認識表達式樹 為什麼要這樣說?它和委托有 ...
  • 在使用Django等框架來操作MySQL時,實際上底層還是通過Python來操作的,首先需要安裝一個驅動程式,在Python3中,驅動程式有多種選擇,比如有pymysql以及mysqlclient等。使用pip命令安裝mysqlclient失敗應如何解決? 安裝的python版本說明 機器同時安裝了 ...