C#串口通訊概念及簡單的實現(xiàn)方法
前言
最近在研究串口通訊,其中有幾個比較重要的概念,RS-232這種適配于上位機(jī)和PC端進(jìn)行連接,RS-232只限于PC串口和設(shè)備間點對點的通信。它很簡單的就可以進(jìn)行連接,由于串口通訊是異步的,也就是說你可以同時向兩端或者更多進(jìn)行數(shù)據(jù)發(fā)送,它們之間的傳輸數(shù)據(jù)類型是byte,串口通信最重要的參數(shù)是波特率、數(shù)據(jù)位、停止位和奇偶校驗。對于兩個進(jìn)行通信的端口,這些參數(shù)必須匹配。
聽大佬說的幾個關(guān)于串口通訊的術(shù)語,啥?啥,這是啥?
就讓我這個“小白”給你說說:第一個波特率,這個東西在不同領(lǐng)域都有涉及,在串口通訊中兩個串口之間通訊之間的信號單元叫做碼元,每分鐘傳遞的信號(碼元)也就是波特率;
第二個數(shù)據(jù)位、停止位:這個聽名字就知道是啥意思了,在傳遞數(shù)據(jù)過程之前,機(jī)器會識別你的數(shù)據(jù)是個啥,然后根據(jù)這個類型,去調(diào)整不同的起始位、停止位。
第三個奇偶校驗:就是想知道你這個數(shù)據(jù)有多大,多長,因為傳輸都是1 和 0 ,具體校驗方法如下:
奇校驗:就是讓原有數(shù)據(jù)序列中(包括你要加上的一位)1的個數(shù)為奇數(shù) ---- 1000110(0)你必須添0這樣原來有3個1已經(jīng)是奇數(shù)了所以你添上0之后1的個數(shù)還是奇數(shù)個。
偶校驗:就是讓原有數(shù)據(jù)序列中(包括你要加上的一位)1的個數(shù)為偶數(shù) ---- 1000110(1)你就必須加1了這樣原來有3個1要想1的個數(shù)為偶數(shù)就只能添1了。
如果想要在C#中玩轉(zhuǎn)串口通訊或者模擬串口,你必須先搞個硬件,當(dāng)然我們是在模擬,那你就弄個模擬的工具唄,這個我給大家準(zhǔn)備好了,自己去下載吧
鏈接: https://pan.baidu.com/s/1kXIN2s__KGGT5EkckWbPhQ 提取碼: 85xh
安裝好了之后,桌面會有圖標(biāo),然后打開,添加一個串口,具體請看下圖
下面我們就開始代碼的實現(xiàn)了,創(chuàng)建一個.NET Fwk的 Winform 項目,然后開始我們的代碼!
SerialPort sp1 = new SerialPort(); public Form1() { InitializeComponent(); } private void Form1_Load(object sender, EventArgs e) { string[] strCom = SerialPort.GetPortNames(); if (strCom == null) { MessageBox.Show("本機(jī)沒有串口!", "Error"); return; } foreach (string com in System.IO.Ports.SerialPort.GetPortNames()) { cbCom.Items.Add(com); } cbCom.SelectedIndex = 0; sp1.BaudRate = 9600; Control.CheckForIllegalCrossThreadCalls = false; sp1.DataReceived += Sp1_DataReceived; sp1.DtrEnable = true; sp1.RtsEnable = true; sp1.ReadTimeout = 1000; sp1.Close(); }
在窗體加載中,我們嘗試獲取了本機(jī)的所有串口,通過 SerialPort.GetPortNames() 方法,會返回一個字符串?dāng)?shù)組,如果沒有就返回null,在其中,我們還設(shè)置了波特率,通過SerialPort類下的 DataReceived 方法,來監(jiān)聽我們的數(shù)據(jù)回傳,還啟用了DTR\DTS請求方式,通過 ReadTimeOut 對數(shù)據(jù)讀取超時進(jìn)行了控制?!?/p>
既然我們監(jiān)聽了數(shù)據(jù)回傳,那么在這個方法中,通過 ReadTimeOut.Read 進(jìn)行了讀取,直接從0到最后,截取完畢。
private void Sp1_DataReceived(object sender, SerialDataReceivedEventArgs e) { if (sp1.IsOpen) //判斷是否打開串口 { //輸出當(dāng)前時間 DateTime dt = DateTime.Now; txtReceived.Text += dt.GetDateTimeFormats('f')[0].ToString() + "\r\n"; Byte[] receivedData = new Byte[sp1.BytesToRead]; //創(chuàng)建接收字節(jié)數(shù)組 sp1.Read(receivedData, 0, receivedData.Length); //讀取數(shù)據(jù) AddContent(new UTF8Encoding().GetString(receivedData)); } else { MessageBox.Show("請打開某個串口", "錯誤提示"); } } private void AddContent(string v) { this.BeginInvoke(new MethodInvoker(delegate { txtReceived.AppendText(v); txtReceived.AppendText("\r\n"); })); }
當(dāng)然以上的操作都必須在打開串口之后才能進(jìn)行,打開串口這個還是比較費勁的,因為要設(shè)置各種參數(shù),也就是我剛才說的那幾個概念,如果不對,你讓其他的端口來訪問就找不到了。
private void button2_Click(object sender, EventArgs e) { if (!sp1.IsOpen) { try { string serialName = cbCom.SelectedItem.ToString(); sp1.PortName = serialName; string strBaudRate = cbBaudRate.Text; string strDateBits = cbDataBits.Text; string strStopBits = cbStop.Text; Int32 iBaudRate = Convert.ToInt32(strBaudRate); Int32 iDateBits = Convert.ToInt32(strDateBits); sp1.BaudRate = iBaudRate; //波特率 sp1.DataBits = iDateBits; //數(shù)據(jù)位 switch (cbStop.Text) //停止位 { case "1": sp1.StopBits = StopBits.One; break; case "1.5": sp1.StopBits = StopBits.OnePointFive; break; case "2": sp1.StopBits = StopBits.Two; break; default: MessageBox.Show("Error:參數(shù)不正確!", "Error"); break; } switch (cbCheck.Text) //校驗位 { case "無": sp1.Parity = Parity.None; break; case "奇校驗": sp1.Parity = Parity.Odd; break; case "偶校驗": sp1.Parity = Parity.Even; break; default: MessageBox.Show("Error:參數(shù)不正確!", "Error"); break; } if (sp1.IsOpen == true) { sp1.Close(); } //設(shè)置必要控件不可用 cbCom.Enabled = false; cbBaudRate.Enabled = false; cbDataBits.Enabled = false; cbStop.Enabled = false; cbCheck.Enabled = false; sp1.Open(); //打開串口 button2.Text = "關(guān)閉串口"; } catch (System.Exception ex) { MessageBox.Show("Error:" + ex.Message, "Error"); return; } } else { cbCom.Enabled = true; cbBaudRate.Enabled = true; cbDataBits.Enabled = true; cbStop.Enabled = true; cbCheck.Enabled = true; sp1.Close(); //關(guān)閉串口 button2.Text = "打開串口"; } }
最后也就是發(fā)送了,剛才我們接收使用Read,發(fā)送也當(dāng)然很簡單,是Write...
private void button1_Click(object sender, EventArgs e) { byte[] sendData = null; if (!sp1.IsOpen) //如果沒打開 { MessageBox.Show("請先打開串口!", "Error"); return; } String strSend = txtSendStr.Text; try { sendData = Encoding.UTF8.GetBytes(txtSendStr.Text.Trim()); sp1.Write(sendData, 0, sendData.Length); } catch (Exception ex) { MessageBox.Show("Error:" + ex.Message, "Error"); } }
還是非常簡單的。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,謝謝大家對我們的支持。
欄 目:C#教程
下一篇:C#創(chuàng)建簡單windows窗體應(yīng)用(加法器)
本文標(biāo)題:C#串口通訊概念及簡單的實現(xiàn)方法
本文地址:http://mengdiqiu.com.cn/a1/C_jiaocheng/4809.html
您可能感興趣的文章
- 01-10C#調(diào)用WebService實例開發(fā)
- 01-10輕松學(xué)習(xí)C#的密封類
- 01-10講解C#面相對象編程中的類與對象的特性與概念
- 01-10C#獲取串口列表實現(xiàn)實時監(jiān)控串口
- 01-10C# 設(shè)計模式系列教程-簡單工廠模式
- 01-10C# 設(shè)計模式系列教程-外觀模式
- 01-10C# 設(shè)計模式系列教程-命令模式
- 01-10C# 設(shè)計模式系列教程-模板方法模式
- 01-10C# 設(shè)計模式系列教程-觀察者模式
- 01-10C# 設(shè)計模式系列教程-適配器模式


閱讀排行
本欄相關(guān)
- 01-10C#通過反射獲取當(dāng)前工程中所有窗體并
- 01-10關(guān)于ASP網(wǎng)頁無法打開的解決方案
- 01-10WinForm限制窗體不能移到屏幕外的方法
- 01-10WinForm繪制圓角的方法
- 01-10C#實現(xiàn)txt定位指定行完整實例
- 01-10WinForm實現(xiàn)仿視頻播放器左下角滾動新
- 01-10C#停止線程的方法
- 01-10C#實現(xiàn)清空回收站的方法
- 01-10C#通過重寫Panel改變邊框顏色與寬度的
- 01-10C#實現(xiàn)讀取注冊表監(jiān)控當(dāng)前操作系統(tǒng)已
隨機(jī)閱讀
- 01-10使用C語言求解撲克牌的順子及n個骰子
- 01-10delphi制作wav文件的方法
- 04-02jquery與jsp,用jquery
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 01-10C#中split用法實例總結(jié)
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 01-11ajax實現(xiàn)頁面的局部加載
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 08-05織夢dedecms什么時候用欄目交叉功能?