C#串口serialPort操作


現在大多數硬件設備均采用串口技術與計算機相連,因此串口的應用程序開發越來越普遍。例如,在計算機沒有安裝網卡的情況下,將本機上的一些信息數據 傳輸到另一台計算機上,那么利用串口通信就可以實現。運行本程序,在“發送數據”文本框中輸入要傳送的數據,單擊【發送】按鈕,將傳送的數據發送到所選擇 的端口號中;單擊【接收】按鈕,傳遞的數據被接收到“接收數據”文本框中。如圖13.1所示。 
  技術要點
  在.NET Framework 2.0中提供了SerialPort類,該類主要實現串口數據通信等。下面主要介紹該類的主要屬性(表13.1)和方法(表13.2)。
  表13.1 SerialPort類的常用屬性
  名稱
  說明
  BaseStream
  獲取 SerialPort 對象的基礎 Stream 對象
  BaudRate
  獲取或設置串行波特率
  BreakState
  獲取或設置中斷信號狀態
  BytesToRead
  獲取接收緩沖區中數據的字節數
  BytesToWrite
  獲取發送緩沖區中數據的字節數
  CDHolding
  獲取端口的載波檢測行的狀態
  CtsHolding
  獲取“可以發送”行的狀態
  DataBits
  獲取或設置每個字節的標准數據位長度
  DiscardNull
  獲取或設置一個值,該值指示 Null 字節在端口和接收緩沖區之間傳輸時是否被忽略
  DsrHolding
  獲取數據設置就緒 (DSR) 信號的狀態
  DtrEnable
  獲取或設置一個值,該值在串行通信過程中啟用數據終端就緒 (DTR) 信號
  Encoding
  獲取或設置傳輸前后文本轉換的字節編碼
  Handshake
  獲取或設置串行端口數據傳輸的握手協議
  IsOpen
  獲取一個值,該值指示 SerialPort 對象的打開或關閉狀態
  NewLine
  獲取或設置用於解釋 ReadLine( )和WriteLine( )方法調用結束的值
  Parity
  獲取或設置奇偶校驗檢查協議
  
  續表
  名稱
  說明
  ParityReplace
  獲取或設置一個字節,該字節在發生奇偶校驗錯誤時替換數據流中的無效字節
  PortName
  獲取或設置通信端口,包括但不限於所有可用的 COM 端口
  ReadBufferSize
  獲取或設置 SerialPort 輸入緩沖區的大小
  ReadTimeout
  獲取或設置讀取操作未完成時發生超時之前的毫秒數
  ReceivedBytesThreshold
  獲取或設置 DataReceived 事件發生前內部輸入緩沖區中的字節數
  RtsEnable
  獲取或設置一個值,該值指示在串行通信中是否啟用請求發送 (RTS) 信號
  StopBits
  獲取或設置每個字節的標准停止位數
  WriteBufferSize
  獲取或設置串行端口輸出緩沖區的大小
  WriteTimeout
  獲取或設置寫入操作未完成時發生超時之前的毫秒數
  
  表13.2 SerialPort類的常用方法
  方 法 名 稱
  說明
  Close
  關閉端口連接,將 IsOpen 屬性設置為False,並釋放內部 Stream 對象
  Open
  打開一個新的串行端口連接
  Read
  從 SerialPort 輸入緩沖區中讀取
  ReadByte
  從 SerialPort 輸入緩沖區中同步讀取一個字節
  ReadChar
  從 SerialPort 輸入緩沖區中同步讀取一個字符
  ReadLine
  一直讀取到輸入緩沖區中的 NewLine 值
  ReadTo
  一直讀取到輸入緩沖區中指定 value 的字符串
  Write
  已重載。將數據寫入串行端口輸出緩沖區
  WriteLine
  將指定的字符串和 NewLine 值寫入輸出緩沖區
  
  注意:用跳線使串口的第2、3針連接,可以在本地計算機上實現串口通信,所以,通過串口的第2、3針的連接可以對程序進行檢測。串口截面圖如圖13.2所示。
  圖13.2 串口截面圖
  實現過程
  (1)新建一個項目,命名為Ex13_01,默認窗體為Form1。
  (2)在Form1窗體中,主要添加兩個Button控件,分別用於執行發送數據和接受數據,添加兩個TextBox控件,用於輸入發送數據和顯示接收數據。
  (3)主要程序代碼。
   private void button1_Click(object sender, EventArgs e)
   {
   serialPort1.PortName = "COM1";
   serialPort1.BaudRate = 9600;
   serialPort1.Open();
   byte[] data = Encoding.Unicode.GetBytes(textBox1.Text);
   string str = Convert.ToBase64String(data);
   serialPort1.WriteLine(str);
   MessageBox.Show("數據發送成功!","系統提示");
   }
   private void button2_Click(object sender, EventArgs e)
   {
   byte[] data = Convert.FromBase64String(serialPort1.ReadLine());
   textBox2.Text = Encoding.Unicode.GetString(data);
   serialPort1.Close();
   MessageBox.Show("數據接收成功!","系統提示");
   }
  舉一反三
  根據本實例,讀者可以實現以下功能。
  遠程監控對方計算機屏幕。
   下位機控制程序。
  實例419通過串口關閉對方計算機
  實例說明
  在網絡應用程序中,主要通過網卡實現數據的傳輸,因此可以利用套接字技術實現遠程關閉計算機。如果計算機中沒有安裝網卡,該如何實現遠程關閉計算機呢?本例實現了利用串口關閉對方計算機,程序運行結果如圖13.3所示。
  技術要點
  本實例使用SerialPort類的屬性和方法,請參見實例“通過串口發送數據”。下面主要介紹SerialPort類的DataReceived 事件,DataReceived 事件為本實例的主要使用技術。DataReceived事件表示將處理 SerialPort 對象的數據接收事件的方法。串行接收事件可以由 SerialData 枚舉中的任何項引起,是否引發此事件由操作系統決定,所以不一定會報告所有奇偶校驗錯誤。
  注意:本實例從開發到測試,都是由本地計算機完成的,用戶只需要使用跳線將串口的第2、3針連接,可以在本地計算機上實現串口通信。跳線連接請參見圖13.2。
  實現過程
  (1)新建一個項目,命名為Ex13_02,默認窗體為Form1。
  (2)在Form1窗體中,主要添加兩個Button控件,分別用於打開通信串口和關閉對方計算機。
  (3)主要程序代碼。
   private void button1_Click(object sender, EventArgs e)
   {
   //打開串口
   serialPort1.PortName = "COM1";
   serialPort1.Open();
   button1.Enabled = false;
   button2.Enabled = true;
   } //數據接收事件,等待接收關機命令
   private void serialPort1_DataReceived(object sender, SerialDataReceivedEventArgs e)
   {
   byte[] data = Convert.FromBase64String(serialPort1.ReadLine());
   string str = Encoding.Unicode.GetString(data);
   serialPort1.Close();
   if (str == "關機")
   {
   Process p = new Process();
   p.StartInfo.FileName = "cmd.exe";
   p.StartInfo.UseShellExecute = false;
   p.StartInfo.RedirectStandardInput = true;
   p.StartInfo.RedirectStandardOutput = true;
   p.StartInfo.RedirectStandardError = true;
   p.StartInfo.CreateNoWindow = true;
   p.Start();
   p.StandardInput.WriteLine("shutdown /s");
   p.StandardInput.WriteLine("exit");
   }
   } //發送關機命令
   private void button2_Click(object sender, EventArgs e)
   {
   if (button2.Text == "關閉計算機")
   {
   //發送關機命令數據
   byte[] data = Encoding.Unicode.GetBytes("關機");
   string str = Convert.ToBase64String(data);
   serialPort1.WriteLine(str);
   button2.Text = "取消關機";
   }
   else
   {
   button2.Text = "關閉計算機";
   button1.Enabled = true;
   button2.Enabled = false;
   //取消關機
   Process p = new Process();
   p.StartInfo.FileName = "cmd.exe";
   p.StartInfo.UseShellExecute = false;
   p.StartInfo.RedirectStandardInput = true;
   p.StartInfo.RedirectStandardOutput = true;
   p.StartInfo.RedirectStandardError = true;
   p.StartInfo.CreateNoWindow = true;
   p.Start();
   p.StandardInput.WriteLine("shutdown /a");
   p.StandardInput.WriteLine("exit");
   }
   }
  在我的測試軟件中發現一個問題,就是當發送數據小於或等於8位時,一切正常,如果大於8為字節,則在datareceived事件中接收到的數據會分成兩段,第一段為8位,第二段為剩下的字節,很奇怪,在msdn中講到不能保證每次發送的數據都能正確接收到,需要參照BytesToRead屬性來確定要讀取的數據量,所以我想出來的解決辦法為:
  int DataLength=serialPort.BytesToRead;
  int i=0;
  StringBuilder sb=new StringBuilder();
  while(i<DataLength)
  {
   byte[] ds=new byte[1024];
   int len=serialPort.Read(ds,0,1024);
   sb.Append(Encoding.Ascii.GetString(ds,0,len));
   i+=len;
  }
  Console.Write(sb,ToString());
  這種奇怪的方法可以解決問題,后來想想應該是串口的工作方式決定的也有可能,期待其他的解決方式。

 

 

 

 微軟代碼:取出本機的COM端口字符串

public static string[] GetPortNames()

{

    RegistryKey localMachine = null;

    RegistryKey key2 = null;

string[] textArray = null;

//這里有個斷言,判斷該注冊表項是否存在

    new RegistryPermission(RegistryPermissionAccess.Read, @"HKEY_LOCAL_MACHINE\HARDWARE\DEVICEMAP\SERIALCOMM").Assert();

    try

    {

        localMachine = Registry.LocalMachine;

        key2 = localMachine.OpenSubKey(@"HARDWARE\DEVICEMAP\SERIALCOMM", false);

        if (key2 != null)

        {

            string[] valueNames = key2.GetValueNames();

            textArray = new string[valueNames.Length];

            for (int i = 0; i < valueNames.Length; i++)

            {

                textArray[i] = (string) key2.GetValue(valueNames[i]);

            }

        }

    }

    finally

    {

        if (localMachine != null)

        {

            localMachine.Close();

        }

        if (key2 != null)

        {

            key2.Close();

        }

        CodeAccessPermission.RevertAssert();

    }

    if (textArray == null)

    {

        textArray = new string[0];

    }

    return textArray;

}

 

 

 

 

VS.NET2005中SerialPort控件操作詳解(C#)

using System;

using System.Collections.Generic;

using System.ComponentModel;

using System.Data;

using System.Drawing;

using System.Text;

using System.Windows.Forms;

using System.IO.Ports;

namespace SerialPorts

{

    public partial class frm_Main : Form

    {

        #region Public Enumerations

        public enum DataMode { Text, Hex }

        public enum LogMsgType { Incoming, Outgoing, Normal, Warning, Error };

        #endregion

        private Color[] LogMsgTypeColor = { Color.Orange, Color.Green, Color.Black, Color.Blue, Color.Red };

        //禁用和啟用程序中各控件的狀態

        private void EnableControls()

        {

            // 基於串口的打開與否,設置控件狀態

            gbPortSettings.Enabled = !ComPort.IsOpen;

            btns.Enabled = btnstop.Enabled = txtSendData.Enabled = btnSend.Enabled = ComPort.IsOpen;

            if (ComPort.IsOpen) btnOpenPort.Text = "關閉串口";

            else btnOpenPort.Text = "打開串口";

        }

        //初始化組件的數據,為串口提供相關參數

        private void InitializeControlValues()

        {

            cmbParity.Items.Clear(); cmbParity.Items.AddRange(Enum.GetNames(typeof(Parity)));

            cmbStopBits.Items.Clear(); cmbStopBits.Items.AddRange(Enum.GetNames(typeof(StopBits)));

            cmbPortName.Items.Clear();

            foreach (string s in SerialPort.GetPortNames())

                cmbPortName.Items.Add(s);

            cmbPortName.Text = cmbPortName.Items[0].ToString();

            cmbParity.Text = cmbParity.Items[0].ToString();

            cmbStopBits.Text = cmbStopBits.Items[0].ToString();

            cmbDataBits.Text = cmbDataBits.Items[0].ToString();

            cmbParity.Text = cmbParity.Items[0].ToString();

            cmbBaudRate.Text = cmbBaudRate.Items[0].ToString();

            EnableControls();

        }

        //十六進制轉換字節數組

        private byte[] HexStringToByteArray(string s)

        {

            s = s.Replace(" ", "");

            byte[] buffer = new byte[s.Length / 2];

            for (int i = 0; i < s.Length; i += 2)

                buffer[i / 2] = (byte)Convert.ToByte(s.Substring(i, 2), 16);

            return buffer;

        }

        //字節數組轉換十六進制

        private string ByteArrayToHexString(byte[] data)

        {

            StringBuilder sb = new StringBuilder(data.Length * 3);

            foreach (byte b in data)

                sb.Append(Convert.ToString(b, 16).PadLeft(2, '0').PadRight(3, ' '));

            return sb.ToString().ToUpper();

        }

        //顯示數據日志

        private void Log(LogMsgType msgtype, string msg)

        {

            rtfTerminal.Invoke(new EventHandler(delegate

            {

                rtfTerminal.SelectedText = string.Empty;

                rtfTerminal.SelectionFont = new Font(rtfTerminal.SelectionFont, FontStyle.Bold);

                rtfTerminal.SelectionColor = LogMsgTypeColor[(int)msgtype];

                rtfTerminal.AppendText(msg);

                rtfTerminal.ScrollToCaret();

            }));

        }

        //串口發送方式

        #region Local Properties

        private DataMode CurrentDataMode

        {

            get

            {

                if (rbHex.Checked) return DataMode.Hex;

                else return DataMode.Text;

            }

            set

            {

                if (value == DataMode.Text) rbText.Checked = true;

                else rbHex.Checked = true;

            }

        }

        #endregion

        //發送數據

        private void SendData()

        {

            if (CurrentDataMode == DataMode.Text)

            {

                // 發送用戶的文本到串口

                ComPort.Write(txtSendData.Text);

                // 將用戶的文本顯示到數據窗口

                Log(LogMsgType.Outgoing, txtSendData.Text + "\n");

            }

            else

            {

                try

                {

                    // 轉換用戶十六進制數據到字節數組

                    byte[] data = HexStringToByteArray(txtSendData.Text);

                    // 發送數據到串口

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

                    // 將用戶十六進制數據到數據窗口

                    Log(LogMsgType.Outgoing, ByteArrayToHexString(data) + "\n");

                }

                catch (FormatException)

                {

                    // 轉換錯誤

                    Log(LogMsgType.Error, "十六進制數據有誤: " + txtSendData.Text + "\n");

                }

            }

            txtSendData.SelectAll();

        }

        /// <summary>

        /// -------------------------------------------------------------

        /// </summary>

        public frm_Main()

        {

            InitializeComponent();

        }

        private void Form1_Load(object sender, EventArgs e)

        {

            InitializeControlValues();

            ComPort.DataReceived += new SerialDataReceivedEventHandler(port_DataReceived);

        }

        //打開串口

        private void btnOpenPort_Click(object sender, EventArgs e)

        {

            if (ComPort.IsOpen) ComPort.Close();

            else

            {

                //設置串口參數

                ComPort.BaudRate = int.Parse(cmbBaudRate.Text);

                ComPort.DataBits = int.Parse(cmbDataBits.Text);

                ComPort.StopBits = (StopBits)Enum.Parse(typeof(StopBits), cmbStopBits.Text);

                ComPort.Parity = (Parity)Enum.Parse(typeof(Parity), cmbParity.Text);

                ComPort.PortName = cmbPortName.Text;

                // 打開串口

                ComPort.Open();

            }

            // 改變組件狀態

            EnableControls();

            // 如果串口打開,將焦點放入txtSendData

            if (ComPort.IsOpen) txtSendData.Focus();

        }

        private void rbHex_CheckedChanged(object sender, EventArgs e)

        {

            if (rbHex.Checked) CurrentDataMode = DataMode.Hex;

        }

        private void rbText_CheckedChanged(object sender, EventArgs e)

        {

            if (rbText.Checked) CurrentDataMode = DataMode.Text;

        }

        //接收數據

        private void port_DataReceived(object sender, SerialDataReceivedEventArgs e)

        {

            

            // 判斷用戶用的是字節模式還是字符模式

            if (CurrentDataMode == DataMode.Text)

            {

                // 讀取緩沖區的數據

                string data = ComPort.ReadExisting();

                // 顯示讀取的數據到數據窗口

                Log(LogMsgType.Incoming, data + "\n");

            }

            else

            {

                // 獲取字節長度

                int bytes = ComPort.BytesToRead;

                // 創建字節數組

                byte[] buffer = new byte[bytes];

                // 讀取緩沖區的數據到數組

                ComPort.Read(buffer, 0, bytes);

                // 顯示讀取的數據到數據窗口

                Log(LogMsgType.Incoming, ByteArrayToHexString(buffer) + "\n");

            }

        }

        //發送數據按鍵

        private void btnSend_Click(object sender, EventArgs e)

        {

            SendData();

        }

        private void lnkAbout_LinkClicked(object sender, LinkLabelLinkClickedEventArgs e)

        {

            (new frmAbout()).ShowDialog(this);

        }

        //時間組件控制發送數據

        private void timer1_Tick(object sender, EventArgs e)

        {

            SendData();

        }

        //連續發送數據

        private void button1_Click(object sender, EventArgs e)

        {

            delay.Enabled = true;

            btns.Enabled = !delay.Enabled;

            btnstop.Enabled = delay.Enabled;

        }

        //停止連續發送數據

        private void button2_Click(object sender, EventArgs e)

        {

            delay.Enabled = false;

            btns.Enabled = !delay.Enabled;

            btnstop.Enabled = delay.Enabled;

        }

    }

}

 原文鏈接:http://blog.csdn.net/jsjpanxiaoyu/article/details/42512207


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM