C#作一个简单的进行串口通讯的上位机

一、上位机与下位机

        上位机至关于一个软件系统,能够用于接收数据、控制数据。便可以对接收到的数据直接发送操控命令来操做数据。上位机能够接收下位机的信号。下位机是一个控制器,是直接控制设备获取设备情况的计算机。上位机发出的命令首先给下位机,下位机再根据此命令解释成相应时序信号直接控制相应设备。下位机不时读取设备状态数据(通常为模拟量),转换成数字信号反馈给上位机。上位机不能够单独使用,而下位机能够单独使用。前端

 

二、串口通讯

        串口至关于硬件类型的接口。好比无线传感节点发送信号到汇聚节点,汇聚节点经过串口将数据传到计算机中的上位机中,上位机接收信息,并处理。编程

      串口是按位(bit)发送和接收字节。串口通讯最重要的参数是波特率、数据位、中止位和奇偶校验。对于两个进行通讯的端口,这些参数必须匹配。c#

    a,波特率:这是一个衡量符号传输速率的参数。后端

    b,数据位:这是衡量通讯中实际数据位的参数。数组

    c,中止位:用于表示单个包的最后一位。典型的值为1,1.5和2位。函数

    d,奇偶校验位:在串口通讯中一种简单的检错方式。测试

 

三、C#代码

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
using System.IO.Ports;
using System.Diagnostics;
namespace serial2
{
    public partial class Form1 : Form
    {
        SerialPort s = new SerialPort();    //实例化一个串口对象,在前端控件中能够直接拖过来,但最好是在后端代码中写代码,这样复制到其余地方不会出错。s是一个串口的句柄
        public Form1()
        {
            InitializeComponent();
            Control.CheckForIllegalCrossThreadCalls = false;   //防止跨线程访问出错,好多地方会用到
            button1.Text = "打开串口";
            int[] item = { 9600,115200};    //定义一个Item数组,遍历item中每个变量a,增长到comboBox2的列表中
            foreach (int a in item)
            {
                comboBox2.Items.Add(a.ToString());
            }
           
            comboBox2.SelectedItem = comboBox2.Items[1];    //默认为列表第二个变量
        }
        private void Form1_Load(object sender, EventArgs e)   //窗体事件要先配置端口信息。
        {
            string[] ports = SerialPort.GetPortNames();
            comboBox1.Items.AddRange(ports);
            comboBox1.SelectedItem=comboBox1.Items[0];
            //Array.Sort(ports);
            
        }
        private void button1_Click(object sender, EventArgs e)   //下面讲解中差很少已经讲清楚了
        {
            try
            {
                if (!s.IsOpen)
                {
                    s.PortName = comboBox1.SelectedItem.ToString();
                    s.BaudRate = Convert.ToInt32(comboBox2.SelectedItem.ToString());
                    s.Open();
                    s.DataReceived += s_DataReceived;
                    button1.Text = "关闭串口";
                    //MessageBox.Show("串口已打开");
                }
                else
                {
                    s.Close();
                    s.DataReceived -= s_DataReceived;
                    button1.Text = "打开串口";
                }
            }
            catch (Exception ee)
            {
                MessageBox.Show(ee.ToString());
            }
        }
        void s_DataReceived(object sender, SerialDataReceivedEventArgs e)   //数据接收事件,读到数据的长度赋值给count,若是是8位(节点内部编程规定好的),就申请一个byte类型的buff数组,s句柄来读数据
        {
            int count =s.BytesToRead;    
            string str=null ;
            if (count == 8)
            {
                byte[] buff = new byte[count];
                s.Read(buff, 0, count);
                foreach (byte item in buff)    //读取Buff中存的数据,转换成显示的十六进制数
                {
                    str += item.ToString("X2")+" ";
                }
                richTextBox1.Text =System.DateTime.Now.ToString()+": "+ str + "\n" + richTextBox1.Text;      //这是跨线程访问richtextbox,原程序和DataReceived事件是两个不一样的线程同时在执行
                if (buff[0] == 0x04)   //若是节点是04发来的数据
                {
                    ID.Text = buff[0].ToString();   //这下面是上位机右边那一段,用来显示处理好的数据的温度、湿度、光照、灰尘、ID信息的。buff【0】中存的是数据的ID信息,显示在ID的Label上面
                    switch (buff[2])   //判断数据类型  buff【0】和buff【1】表明ID的低位和高位,同理2和3表明数据类型的低位和高位,当2和3的值为1时,4和5表明温度,6和7表明湿度;
                    
         {  
                      case 0x01:       //当2和3的值为1,4和5是温度,6和7是湿度
          {
                                Tem.Text = (buff[5] * 4 + buff[4] * 0.05 - 30).ToString();
                                Hum.Text = (buff[6]  + buff[7]).ToString();
                                break;
                            }
                        case 0x02://6和7是光照
                       {
                                Light.Text = (buff[6] + buff[7]).ToString();
                                break;
                            }
                        case 0x04://6和7是灰尘
                        {
                                Dust.Text = (buff[6] + buff[7]).ToString();
                                break;
                            }
                        default:
                            break;
                    }
                }
            
            }
            
        }
        private void button3_Click(object sender, EventArgs e)   //每次发一个字节
   {
            string[] sendbuff = richTextBox2.Text.Split();  //分割输入的字符串,判断有多少个字节须要发送
        Debug.WriteLine("发送字节数:"+sendbuff.Length);
            foreach (string  item in sendbuff)
            {
                int count = 1;
                byte[] buff = new byte[count];
                buff[0] = byte.Parse(item, System.Globalization.NumberStyles.HexNumber);//格式化字符串为十六进制数值
              s.Write(buff, 0, count);
            }
        }
        private void button2_Click(object sender, EventArgs e)//刷新右边的数值
      {
            int count = 1;
            byte[] buff = new byte[count];
            buff[0] = byte.Parse("04", System.Globalization.NumberStyles.HexNumber);//这里只显示04节点的信息
        s.Write(buff, 0, count);
        }
    }
}

 (以上规则均是本实验室节点内部自定义规则,测试的,外面的相应要改)spa

四、结果

 

 

五、补充四点知识

  1)在程序可能会遇到错误的地方,用try+两个Tab键,将代码写入try中。好比本例子中的代码:线程

 private void button1_Click(object sender, EventArgs e)
        {
            try
            {
                if (!s.IsOpen)
                {
                    s.PortName = comboBox1.SelectedItem.ToString();
                    s.BaudRate = Convert.ToInt32(comboBox2.SelectedItem.ToString());
                    s.Open();
                    s.DataReceived += s_DataReceived;
                    button1.Text = "关闭串口";
                    //MessageBox.Show("串口已打开");
                }
                else
                {
                    s.Close();
                    s.DataReceived -= s_DataReceived;
                    button1.Text = "打开串口";
                }
            }
            catch (Exception ee)
            {
                MessageBox.Show(ee.ToString());
            }
        }

若是代码没有写入try中,则可能出现的一种状况是好比有两个上位机,同时占用同一个串口,则就会冲突,会出错。程序就会终止,整个进程结束。而若是写入try中,而且把抛出异常的catch代码实例化,即捕获异常要实例化一个句柄,这样程序遇到error就不会终止,而会出现报错的缘由。以下图,个人这个上位机和网上下载的一个上位机同时占用COM3串口(网上下载的先占用COM3),这时个人上位机在打开串口时会出现报错。code

2)就我这个上位机而言,须要有打开串口和关闭串口两个button按钮,可是考虑到占地方,固然最重要的仍是若是用两个按钮来表示,当你按下打开串口,若是忘了是否打开,则是看不出来是否是打开的,因此能够合并为一个button控件。(代码仍是用上面那一段的代码)。(感受很神奇啊)。在button1_Click事件中,先点击button,若是串口是关闭的,则打开串口,而后把button1.Text的值赋值为“关闭串口”,若是串口原本是关闭的,则点击按钮会把button1.Text的值赋值为“打开串口”,同时把接收的数据清空。感受这个方法真的很不错!嘿嘿

3)当输入一个变量或方法什么的,它全部有的会自动出如今一个列表,这时,“正方体”表明“方法”,“小钳子”表明“变量”,“闪电”表明“事件”。

4) 产生对象的事件时

好比输入s.自动会出现DataReceived事件,再输入“+=”就会有如上图提示,按Tab键。而后又会以下图提示

再次按tab键,就会自动生成DataReceived事件处理函数。

相关文章
相关标签/搜索