C#中如何安全的关闭串口

VC#中如果涉及到多线程,特别是大量的数据处理和界面更新时,如果简单强制的关闭串口,很可能会造成串口死掉。
串口无法关闭的原因是:要关闭串口的时候,有其它线程还在读取数据或者更新界面。
关键是:在准备关闭串口的时候,看看是否在接收和处理数据,如果是就等它处理完为止;在事件处理的最前面,判断如果是准备关闭串口的bool类型值,就不再进入数据接收和处理。

using System;
using System.Collections.Generic;
using System.Text;
using System.IO.Ports;
using System.Timers;


namespace PortTesting
{
    /// <summary>
    /// 定义了一个委托类型
    /// </summary>
    public delegate void WhenGetNew();

    /// <summary>
    /// 串口封装类,Help By Wyz
    /// </summary>
    public class PortDataDisplay
    {
       /// <summary>
       /// 系统串口类
       /// </summary>
       public SerialPort serialPort = new SerialPort("COM1", 19200);
       /// <summary>
       /// 解析得到数据后触发事件
       /// </summary>
       public event WhenGetNew whenGetNew;
       /// <summary>
       /// 处理线程
       /// </summary>
       private SerialDataReceivedEventHandler threadCallHandler;
       /// <summary>
       /// 对外的数据类型定义
       /// </summary>
       public string dataSrc = "";
       /// <summary>
       /// 准备关闭串口=true
       /// </summary>
       private bool m_IsTryToClosePort = false;
       /// <summary>
       /// true表示正在接收数据
       /// </summary>
       private bool m_IsReceiving = false;
       

       /// <summary>
       /// 初始化
       /// </summary>
       public PortDataDisplay()
       {
       }

       /// <summary>
       /// 有参数的构造函数
       /// </summary>
       /// <param name="PortName">串口号,如"COM1"</param>
       /// <param name="BaudRate">波特率,如19200</param>
       public PortDataDisplay(string PortName, int BaudRate)
       {
           serialPort = new SerialPort(PortName, BaudRate);
       }

       /// <summary>
       /// 开始工作
       /// </summary>
       public void ConnectDeveice()
       {
           //0.注册事件
           serialPort.DataReceived -= OnSerialPortDataCome;
           serialPort.DataReceived += OnSerialPortDataCome;
           //1.再设置一下串口参数
           if (this.serialPort.IsOpen == false)
           {
               this.serialPort.ReadBufferSize = 1000;
               this.serialPort.ReceivedBytesThreshold = 1;//数据达到120的时候才就要触发事件,不行!!应该是数据来就触发
               //2.打开串口开始工作
               m_IsTryToClosePort = false;
               this.serialPort.Open();
           }
       }

       /// <summary>
       /// 结束工作
       /// </summary>
       public void DisconnectDeveice() // 关键和核心!!!
       {
           m_IsTryToClosePort = true;
           while (m_IsReceiving)
           {
               System.Windows.Forms.Application.DoEvents();
           }
           serialPort.Close();
       }

       /// <summary>
       /// 当通知到有数据达到120时处理(读取,与分析)
       /// </summary>
       /// <param name="sender"></param>
       /// <param name="e"></param>
       private void OnSerialPortDataCome(object sender, SerialDataReceivedEventArgs e)
       {
           if (m_IsTryToClosePort) // 关键!!!
           {
               return;
           }

           m_IsReceiving = true; // 关键!!!

           try
           {
               if (threadCallHandler == null)
               {
                   threadCallHandler = new SerialDataReceivedEventHandler(OnSerialPortDataCome);
               }

               //read
               dataSrc = serialPort.ReadExisting();//读出缓冲区所有数据
               if (dataSrc != "" && this.whenGetNew != null)
               {
                   this.whenGetNew();
               }
           }
           finally // 放在finally里面比较好。
           {
               m_IsReceiving = false; // 关键!!!
           }
       }
    }
}


 

使用的时候:

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

namespace PortTesting
{
    public partial class FrmMain : Form
    {
       /// <summary>
       /// 封装好的串口类
       /// </summary>
       private PortDataDisplay m_portDispl = new PortDataDisplay("COM1", 19200);

       public FrmMain()
       {
           InitializeComponent();
       }

       private void btnOpen_Click(object sender, EventArgs e)
       {
           if (btnOpen.Text == "打开串口")
           {
               m_portDispl.whenGetNew -= portDispl_whenGetNew;
               m_portDispl.whenGetNew += new WhenGetNew(portDispl_whenGetNew);
               m_portDispl.ConnectDeveice();
               btnOpen.Text = "关闭串口";
           }
           else if (btnOpen.Text == "关闭串口")
           {
               m_portDispl.DisconnectDeveice();
               btnOpen.Text = "打开串口";
           }
       }

       /// <summary>
       /// 事件
       /// </summary>
       private void portDispl_whenGetNew()
       {
           WhenGetNew ehan = delegate
           {
               txtDisplay.AppendText(m_portDispl.dataSrc);
           };

           try
           {
               if (InvokeRequired)
               {
                   this.Invoke(ehan);
               }
           }
           catch
           {
           }
       }


       private void btnClear_Click(object sender, EventArgs e)
       {
           txtDisplay.Clear();
       }
    }
}
  • 3
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值