做工业通 信有很长时间了,特别是串口(232/485),有VB/VC/C各种版本的串口操作代码,这些代码也经过了多年的现场考验,应该说是比较健壮的代码,但 是目前却没有C#相对成熟的串口操作代码,最近用Moxa的设备开发基于WinCE5.0的串口操作代码,所以就扩充完善了一下串口操作,特别是 SendCommand函数,这是我比较常用的主从通信代码,不喜欢用事件或线程接数据,在规定的超时时间内直接循环判断要接收的数据。

     下面是具体的代码:   

  public class PortData

    {

        public event PortDataReceivedEventHandle Received;

        public event SerialErrorReceivedEventHandler Error; 

        public SerialPort port;

        public bool ReceiveEventFlag = false;  //接收事件是否有效 false表示有效


        public PortData(string sPortName, int baudrate,Parity parity,SerialInterface.SerialMode mode)

        {

            port = new SerialPort(sPortName, baudrate, parity, 8, StopBits.One);

            port.RtsEnable = true;

            port.ReadTimeout = 3000;

            port.DataReceived += new SerialDataReceivedEventHandler(DataReceived);

            port.ErrorReceived += new SerialErrorReceivedEventHandler(ErrorEvent);

        }


        ~PortData()

        {

            Close();

        }

        public void Open()

        {

            if (!port.IsOpen)

            {           

                port.Open();

            }

        }


        public void Close()

        {

            if (port.IsOpen)

            {

                port.Close();

            }

        }

        //数据发送

        public void SendData(byte[] data)

        {

            if (port.IsOpen)

            {

                port.Write(data, 0, data.Length);

            }

        }

        public void SendData(byte[] data,int offset,int count)

        {

            if (port.IsOpen)

            {

                port.Write(data, offset, count);

            }

        }

        //发送命令

        public int SendCommand(byte[] SendData, ref  byte[] ReceiveData,int Overtime)

        {


            if(port.IsOpen)

            {

                ReceiveEventFlag = true;        //关闭接收事件

                port.DiscardInBuffer();         //清空接收缓冲区                 

                port.Write(SendData, 0, SendData.Length);

                int num=0,ret=0;

                while (num++ < Overtime)

                {

                    if (port.BytesToRead >= ReceiveData.Length) break;

                    System.Threading.Thread.Sleep(1); 

                }

                if (port.BytesToRead >= ReceiveData.Length) 

                    ret = port.Read(ReceiveData, 0, ReceiveData.Length);

                ReceiveEventFlag = false;       //打开事件

                return ret;

            }

            return -1;

        }


        public void ErrorEvent(object sender, SerialErrorReceivedEventArgs e)

        {

            if (Error != null) Error(sender, e);

        }

        //数据接收

        public void DataReceived(object sender, SerialDataReceivedEventArgs e)

        {

            //禁止接收事件时直接退出

            if (ReceiveEventFlag) return;


            byte[] data = new byte[port.BytesToRead];

            port.Read(data, 0, data.Length);

            if (Received != null) Received(sender, new PortDataReciveEventArgs(data));

        }


        public bool IsOpen()

        {

            return port.IsOpen;

        }

    }

    public delegate void PortDataReceivedEventHandle(object sender, PortDataReciveEventArgs e);

    public class PortDataReciveEventArgs : EventArgs

    {

        public PortDataReciveEventArgs()

        {

            this.data = null;

        }


        public PortDataReciveEventArgs(byte[] data)

        {

            this.data = data;

        }


        private byte[] data;


        public byte[] Data

        {

            get { return data; }

            set { data = value; }

        }

    }

【附注】1~9 串口的名称是 "COMx:",>9的以前用\\\\.\\COMx:比较好使,但是在moxa 661设备上却不行,要用如下格式"$device\\COM" + PortNo.ToString() + "\0",也许这是moxa修改了相应的串口驱动。



//注:把代码中的public PortData(string sPortName, int baudrate,Parity parity,SerialInterface.SerialMode mode) 最后一个参数去掉。

PortData comPort = new PortData("COM1:", 115200, Parity.Even);

byte[] bytSendArray = new byte[2]; //发送数据缓冲区

bytSendArray[0]=0xAC;

bytSendArray[2]=0xAA;

byte[] bytReceiveArray = new byte[5];

//该命令潜台词是你发送了两个字节的数据0xAC,0xAA 下位机应该在200毫秒超时内返回5个字节的数据

intReceiveNum = comPort.SendCommand(bytSendData, ref bytReceiveArray, 200);

//intReceiveNum为实际返回的数据个数,返回的数据放在bytReceiveArray中

comPort.Close();

该代码适合主从式通信(一应一答方式)



 ​



//注:把代码中的public PortData(string sPortName, int baudrate,Parity parity,SerialInterface.SerialMode mode) 最后一个参数去掉。

PortData comPort = new PortData("COM1:", 115200, Parity.Even);

byte[] bytSendArray = new byte[2]; //发送数据缓冲区

bytSendArray[0]=0xAC;

bytSendArray[2]=0xAA;

byte[] bytReceiveArray = new byte[5];

//该命令潜台词是你发送了两个字节的数据0xAC,0xAA 下位机应该在200毫秒超时内返回5个字节的数据

intReceiveNum = comPort.SendCommand(bytSendData, ref bytReceiveArray, 200);

//intReceiveNum为实际返回的数据个数,返回的数据放在bytReceiveArray中

comPort.Close();

该代码适合主从式通信(一应一答方式)