欧美极品高清xxxxhd,国产日产欧美最新,无码AV国产东京热AV无码,国产精品人与动性XXX,国产传媒亚洲综合一区二区,四库影院永久国产精品,毛片免费免费高清视频,福利所导航夜趣136

專注電子技術學習與研究
當前位置:單片機教程網 >> MCU設計實例 >> 瀏覽文章

c#串口通訊軟件設計

作者:劉溫電   來源:本站原創   點擊數:  更新時間:2013年11月25日   【字體:


最近工作用到modbus,同時也用到了上位機軟件,看到主管用vb寫的軟件可以通過串口與我的下位機通訊,顯示數據,覺得很好玩。所以回家一時沖動,就下了個vs2005,準備也搞一搞,但用個啥語言呢?思考了一下,vb--太老,c++,太復雜,java,貌似得加軟件,沒敢玩,最后思來想去,決定了一個比較靠普的語言,c#,這個目前寫軟件還是比較流行的,用的人很多,例子也很多,關鍵是類似vb,但是以c為基礎的。所以很適合我。下面是通過了1天艱苦奮斗,寫的一個破串口通訊軟件 
  
 

調試的截屏,有點傻,不過基本功能還是可以的,計數,發送,接受,串口,波特率設置,都是可以設置的,不過校驗目前還沒有,第一步算是搞定了,以后慢慢完善。

發現這東西,其實也不太難。很多東西都用函數封裝起來了,直接調用就行,貌似比c語言方便。

創建項目就不說了,寫com代碼事,需要加一些類似的頭文件的聲明。

using System.IO.Ports;      

在界面初始化時,初始化串口

private SerialPort comm = new SerialPort();    //具體干啥的,我也不知道 貌似是將comm變成串口結構

這個語句很有意思,我局的很神奇,就是搜索串口號,然后將其復制在ports數組中。

string[] ports = SerialPort.GetPortNames();     //獲取串口號,將其存入一個一維字符串數組

這樣的話,一下子,就可以知道目前哪個串口是處于可工作的狀態。

Array.Sort(ports);                              //對一個一維數組進行排列

這個函數也挺有意思,他可以對字符串數組進行排列,一句話就搞定了,如果是c的話,費了勁了,首先要將字符串變成數據,然后再對數據進行排列,至于怎么樣排列,我也不知道,沒做過,可是這個,一句就可以了,不錯不錯

labelTXdata.Text = "發送數據:" + send_count.ToString();

上面這句,可以使文字控件按照你的意思顯示變量,類似在c語言中,讓1602顯示數據一樣。

void comm_DataReceived(object sender, SerialDataReceivedEventArgs e)

這個貌似就是串口接受函數,類似c口中斷,接受到數據后,進入這個函數來處理

//依次的拼接出16進制字符串
foreach (byte b in buf)
{
    builder.Append(b.ToString("X2") + " ");
 }

//轉換列表為數組后發送
comm.Write(buf.ToArray(), 0, buf.Count);

在這句話的前面還有那么幾句

//我們不管規則了。如果寫錯了一些,我們允許的,只用正則得到有效的十六進制數
MatchCollection mc = Regex.Matches(SENDBOX.Text, @"(?i)[\da-f]{2}");
List<byte> buf = new List<byte>();//填充到這個臨時列表中
//依次添加到列表中
foreach (Match m in mc)
{
buf.Add(byte.Parse(m.Value, System.Globalization.NumberStyles.HexNumber));
}
//具體干啥用的,我也不知道

//打開串口

comm.Open();

剩下的就是零零碎碎不重要的東西了,下面是全部代碼:有興趣的大小盆友看一下

反正第一次寫軟件,喜樂糊涂的,不管對c

 

 

 

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.IO.Ports;
using System.Windows.Forms;
using System.Text.RegularExpressions;

namespace com
{
    public partial class Form1 : Form
    {

        private SerialPort comm = new SerialPort();
        private StringBuilder builder = new StringBuilder();//避免在事件處理方法中反復的創建,定義到外面。
        private long received_count = 1000;//接收計數
        private long send_count = 1000;//發送計數


        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            string[] ports = SerialPort.GetPortNames();     //獲取串口號,將其存入一個一維字符串數組
            string[] Baudrate = { "4800", "9600", "192000" };
            Array.Sort(ports);                              //對一個一維數組進行排列
            cobportname.Items.AddRange(ports);                //向combobox下拉列表中添加數據,數據為串口號
            cobBaudrate.Items.AddRange(Baudrate);
            cobportname.SelectedIndex = cobportname.Items.Count > 0 ? 0 : -1;             //首選項目
            cobBaudrate.SelectedIndex = cobBaudrate.Items.IndexOf("9600");                //首選項為字符串時
            comm.NewLine = "\r\n";
            comm.RtsEnable = true;//根據實際情況吧。

            //添加事件注冊
            labelTXdata.Text = "發送數據:" + send_count.ToString();
            labelRXdata.Text = "接受數據:" + received_count.ToString();
            comm.DataReceived += comm_DataReceived;
          
        }
        void comm_DataReceived(object sender, SerialDataReceivedEventArgs e)
        {
            int n = comm.BytesToRead;//先記錄下來,避免某種原因,人為的原因,操作幾次之間時間長,緩存不一致
            byte[] buf = new byte[n];//聲明一個臨時數組存儲當前來的串口數據
            received_count += n;//增加接收計數
            comm.Read(buf, 0, n);//讀取緩沖數據
            builder.Remove(0, builder.Length);//清除字符串構造器的內容
            this.Invoke((EventHandler)(delegate
            {
                //判斷是否是顯示為16禁止
                if(HEXRX.Checked)
                {
                    //依次的拼接出16進制字符串
                    foreach (byte b in buf)
                    {
                        builder.Append(b.ToString("X2") + " ");
                    }
                }
                else
                {
                    //直接按ASCII規則轉換成字符串
                    builder.Append(Encoding.ASCII.GetString(buf));
                }
                //追加的形式添加到文本框末端,并滾動到最后。
                this.GETRX.AppendText(builder.ToString());
                //修改接收計數
                labelTXdata.Text = "發送數據:" + send_count.ToString();
                labelRXdata.Text = "接受數據:" + received_count.ToString();
              
            }));
        }
        private void buttonTX_Click(object sender, EventArgs e)
        {
            //定義一個變量,記錄發送了幾個字節
            int n = 0;
            //16進制發送
            if (HEXTX.Checked)
            {
                //我們不管規則了。如果寫錯了一些,我們允許的,只用正則得到有效的十六進制數
                MatchCollection mc = Regex.Matches(SENDBOX.Text, @"(?i)[\da-f]{2}");
                List<byte> buf = new List<byte>();//填充到這個臨時列表中
                //依次添加到列表中
                foreach (Match m in mc)
                {
                    buf.Add(byte.Parse(m.Value, System.Globalization.NumberStyles.HexNumber));
                }
                //轉換列表為數組后發送
                comm.Write(buf.ToArray(), 0, buf.Count);
                //記錄發送的字節數
                n = buf.Count;
            }
            else//ascii編碼直接發送
            {
              
                    comm.WriteLine(SENDBOX.Text);
                    n = SENDBOX.Text.Length + 2;
              
            }
            send_count += n;//累加發送字節數
            labelTXdata.Text = "Send:" + send_count.ToString();//更新界面
        }

        private void groupBox1_Enter(object sender, EventArgs e)
        {

        }

        private void button2_Click(object sender, EventArgs e)
        {
            //復位接受和發送的字節數計數器并更新界面。
            send_count = received_count = 0;
            labelTXdata.Text = "發送數據:" + send_count.ToString();
            labelRXdata.Text = "接受數據:" + received_count.ToString();
        }

        private void bottoncom_Click(object sender, EventArgs e)
        {
            //根據當前串口對象,來判斷操作
            if (comm.IsOpen)
            {
                //打開時點擊,則關閉串口
                comm.Close();
            }
            else
            {
                //關閉時點擊,則設置好端口,波特率后打開
                comm.PortName = cobportname.Text;
                comm.BaudRate = int.Parse(cobBaudrate.Text);
                try
                {
                    comm.Open();
                }
                catch (Exception ex)
                {
                    //捕獲到異常信息,創建一個新的comm對象,之前的不能用了。
                    comm = new SerialPort();
                    //現實異常信息給客戶。
                    MessageBox.Show(ex.Message);
                }
            }
            //設置按鈕的狀態
            bottoncom.Text = comm.IsOpen ? "關閉" : "打開";
            //buttonSend.Enabled = comm.IsOpen;
        }
    }
}
 

關閉窗口

相關文章