| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296 | using Aitex.Core.RT.Event;using Aitex.Core.RT.Log;using Aitex.Core.RT.SCCore;using Aitex.Core.Util;using System;using System.Collections.Concurrent;using System.Collections.Generic;using System.IO.Ports;using System.Linq;using System.Text;using System.Threading;using System.Threading.Tasks;namespace MECF.Framework.Common.Device.ResistivityProbe{    public class ThorntonSerialDevice    {        #region Delegate        public delegate void ThorntonDataChanged(string name, List<string> lstContent);        #endregion        #region 常量        private const string CONSTANT_COMMAND = "D00Z\r";        private const char SPLIT_CHARACTER = '\t';        #endregion        #region 内部变量        /// <summary>        /// 连接状态        /// </summary>        private bool _connected;        /// <summary>        /// 串口        /// </summary>        private SerialPort _serialPort;        /// <summary>        /// 模块名称        /// </summary>        private string _name;        /// <summary>        /// 定时器        /// </summary>        private PeriodicJob _periodJob;        /// <summary>        /// 接收超时        /// </summary>        private int _receiveTimeout = 2000;        /// <summary>        /// 错误信息        /// </summary>        private string _errmsg = "";        /// <summary>        /// warning信息        /// </summary>        private string _warnMsg = "";        /// <summary>        /// 重连        /// </summary>        private bool _reconnected;        /// <summary>        /// 离线时间        /// </summary>        private DateTime _offlineDateTime;        /// <summary>        /// 首次连接成功        /// </summary>        private bool _isFirstConnected = false;        /// <summary>        /// 锁        /// </summary>        private object _locker = new object();        #endregion        #region 公共变量        public event ThorntonDataChanged OnDataChanged;        #endregion        #region 属性        /// <summary>        /// 连接状态        /// </summary>        public bool Connected        {            get { return _connected; }            set { _connected = value; }        }        #endregion        /// <summary>        /// 构造函数        /// </summary>        /// <param name="name"></param>        /// <param name="portName"></param>        /// <param name="baudRate"></param>        /// <param name="stopBits"></param>        /// <param name="dataBits"></param>        /// <param name="parity"></param>        public ThorntonSerialDevice(string name, string portName, int baudRate = 9600, StopBits stopBits = StopBits.One, int dataBits = 8, Parity parity = Parity.None, bool reconnected = false, int receiveTimeout = 2000)        {            _serialPort = new SerialPort();            _serialPort.BaudRate = baudRate;            _serialPort.StopBits = stopBits;            _serialPort.DataBits = dataBits;            _serialPort.Parity = parity;            _serialPort.PortName = portName;            _serialPort.ErrorReceived += SerialPort_ErrorReceived;            _serialPort.WriteTimeout = receiveTimeout;            _serialPort.ReadTimeout = receiveTimeout;            _receiveTimeout = receiveTimeout;            _name = name;            _reconnected = reconnected;            _periodJob = new PeriodicJob(500, OnTimer, $"{_name}_reader", true, true);        }        /// <summary>        /// 启动        /// </summary>        public void Start()        {            //Open();        }        /// <summary>        /// 打开串口        /// </summary>        private void Open()        {            lock (_locker)            {                if (!_connected)                {                    try                    {                        if (!_serialPort.IsOpen)                        {                            _serialPort.Open();                            LOG.WriteLog(eEvent.INFO_RESISTIVITY, _name, $"connect port[{_serialPort.PortName}] success");                        }                        _connected = true;                    }                    catch (Exception ex)                    {                        WriteErrorMsg(ex.Message);                    }                }            }        }        /// <summary>        /// 关闭        /// </summary>        public void Close()        {            try            {                _connected = false;                _serialPort.Close();                _periodJob.Stop();            }            catch (Exception ex)            {                WriteErrorMsg(ex.Message);            }        }        #region 事件        /// <summary>        /// 串口错误        /// </summary>        /// <param name="sender"></param>        /// <param name="e"></param>        private void SerialPort_ErrorReceived(object sender, SerialErrorReceivedEventArgs e)        {            LOG.WriteLog(eEvent.ERR_RESISTIVITY, _name, e.EventType.ToString());        }        #endregion        /// <summary>        /// 定时器        /// </summary>        /// <returns></returns>        private bool OnTimer()        {            if (!_connected)            {                if (_reconnected)                {                    Open();                }                else                {                    _periodJob.Stop();                }            }            else            {                ReadData();            }            return true;        }        /// <summary>        /// 读取数据        /// </summary>        public void ReadData()        {            string str = "";            try            {                _serialPort.WriteLine(CONSTANT_COMMAND);                DateTime dt = DateTime.Now;                while (DateTime.Now.Subtract(dt).TotalMilliseconds <= _receiveTimeout)                {                    if (_serialPort.BytesToRead > 0)                    {                        string tmp = _serialPort.ReadLine();                        if (!string.IsNullOrEmpty(tmp))                        {                            WriteInfoMsg(0, tmp);                        }                        if (tmp != CONSTANT_COMMAND)                        {                            str = tmp;                            break;                        }                    }                    else                    {                        Thread.Sleep(100);                    }                }                if (!string.IsNullOrEmpty(str))                {                    string[] strAry = str.Split(SPLIT_CHARACTER);                    if (strAry.Length >= 4)                    {                        List<string> lst = new List<string>();                        for (int i = 2; i < strAry.Length; i += 2)                        {                            lst.Add(strAry[i]);                        }                        if (OnDataChanged != null)                        {                            OnDataChanged(_name, lst);                        }                    }                }                _errmsg = "";            }            catch (Exception ex)            {                WriteWarnMsg($"read exception{ex.Message}--{str}");            }        }        /// <summary>        /// 记录错误信息        /// </summary>        /// <param name="msg"></param>        private void WriteErrorMsg(string msg, bool disConnected = true)        {            if (disConnected)            {                _connected = false;                _offlineDateTime = DateTime.Now;            }            if (_errmsg != msg)            {                _errmsg = msg;                LOG.WriteLog(eEvent.ERR_RESISTIVITY, _name, msg);            }        }        /// <summary>        /// 记录错误信息        /// </summary>        /// <param name="msg"></param>        private void WriteWarnMsg(string msg)        {            if (_warnMsg != msg)            {                _warnMsg = msg;                LOG.WriteLog(eEvent.WARN_RESISTIVITY, _name, msg);            }        }        /// <summary>        /// 写日志        /// </summary>        /// <param name="bytes"></param>        private void WriteInfoMsg(int logType, string str)        {            bool enableLog = false;            if (SC.ContainsItem("Log.EnableResistivityLog"))            {                enableLog = SC.GetValue<bool>("Log.EnableResistivityLog");            }            if (enableLog)            {                string type = logType == 0 ? "receive" : "send";                LOG.WriteBackgroundLog(eEvent.INFO_RESISTIVITY, _name, $"{type} {str}");            }        }    }}
 |