一、串口通信协议
1.串口通信协议简介
串口通信(serial communication)
是一种设备间非常常用的串行通信方式,大部分电子设备都支持,电子工程师再调试设备时也经常使用该通信方式输出调试信息。讲到某一种通信协议,离不开的就是物理层,物理层主要表现形式是电平信号的高低幅值,分别代表的状态。 串口的物理层有很多标准,主要包含RS232标准
,该标准规定了信号的用途、通信接口以及信号的电平标准;
在上面的通讯方式中,两个通讯设备的“DB9接口”之间通过串口信号线建立起连接,串口信号线中使用“RS-232标准”传输数据信号。
由于RS-232电平标准的信号不能直接被控制器直接识别,所以这些信号会经过一个“电平转换芯片”转换成控制器能识别的“TTL校准”的电平信号,才能实现通讯。
电平标准
根据通信使用的电平标准不同,串口通信可以分为TTL标准和RS232标准
我们常见的电子电路中使用的TTL电平标准,理想状态下,使用5V表示二级制逻辑1,使用0V表示逻辑0,而为了增加串口通信的远距离传输以及抗干扰能力,它使用-15V表示逻辑1,+15V表示逻辑0,使用RS232与TTL电平标准表示同一个信号时的对比见图
因为控制器一般使用TTL电平标准,所以常常会使用MA3232芯片对TTL以及RS232电平信号进行互相转换。
RS-232信号线 在最初的应用中,RS-232串口标准常用于计算机、路由与调制调解器(MODEN,俗称“猫”)之间的通讯 ,在这种通讯系统中,设备被分为数据终端设备DTE(计算机、路由)和数据通讯设备DCE(调制调解器)。我们以这种通讯模型讲解它们的信号线连接方式及各个信号线的作用。
在旧式的台式计算机中一般会有RS-232标准的COM口(也称DB9接口)
其中接线口以针式引出信号线的称为公头,以孔式引出信号线的称为母头。在计算机中一般引出公头接口,而在调制调解器设备中引出的一般为母头,使用上图中的串口线即可把它与计算机连接起来。通讯时,串口线中传输的信号就是使用前面讲解的RS-232标准调制的。
在这种应用场合下,DB9接口中的公头及母头的各个引脚的标准信号线接法见图
上表中的是计算机端的DB9公头标准接法,由于两个通讯设备之间的收发信号(RXD与TXD)应交叉相连,所以调制调解器端的DB9母头的收发信号接法一般与公头的相反,两个设备之间连接时,只要使用“直通型”的串口线连接起来即可
串口线中的RTS、CTS、DSR、DTR及DCD信号,使用逻辑 1表示信号有效,逻辑0表示信号无效。例如,当计算机端控制DTR信号线表示为逻辑1时,它是为了告知远端的调制调解器,本机已准备好接收数据,0则表示还没准备就绪。
在目前的其它工业控制使用的串口通讯中,一般只使用RXD、TXD以及GND三条信号线,直接传输数据信号。而RTS、CTS、DSR、DTR及DCD信号都被裁剪掉了,如果您在前面被这些信号弄得晕头转向,那就直接忽略它们吧。
协议层
串口通讯的数据包由发送设备通过自身的TXD接口传输到接收设备的RXD接口。在串口通讯的协议层中,规定了数据包的内容,它由启始位、主体数据、校验位以及停止位组成,通讯双方的数据包格式要约定一致才能正常收发数据,其组成见下图。
波特率
在串口异步通讯中由于没有时钟信号(如前面讲解的DB9接口中是没有时钟信号的),所以两个通讯设备之间需要约定好波特率,即每个码元的长度,以便对信号进行解码常见的波特率为4800、9600、115200等。
- 通讯的起始和停止信号 串口通讯的一个数据包从起始信号开始,直到停止信号结束。数据包的起始信号由一个逻辑0的数据位表示,而数据包的停止信号可由0.5、1、1.5或2个逻辑1的数据位表示,只要双方约定一致即可。
- 有效数据 在数据包的起始位之后紧接着的就是要传输的主体数据内容,也称为有效数据,有效数据的长度常被约定为5、6、7或8位长。
- 数据校验 在有效数据之后,有一个可选的数据校验位。由于数据通信相对更容易受到外部干扰导致传输数据出现偏差,可以在传输过程加上校验位来解决这个问题。校验方法有奇校验(odd)、偶校验(even)、0校验(space)、1校验(mark)以及无校验(noparity),它们介绍如下:
奇校验要求有效数据和校验位中“1”的个数为奇数,比如一个8位长的有效数据为:01101001,此时总共有4个“1”,为达到奇校验效果,校验位为“1”,最后传输的数据将是8位的有效数据加上1位的校验位总共9位。
偶校验与奇校验要求刚好相反,要求帧数据和校验位中“1”的个数为偶数,比如数据帧:11001010,此时数据帧“1”的个数为4个,所以偶校验位为“0”。
0校验是不管有效数据中的内容是什么,校验位总为“0”,1校验是校验位总为“1”。
在无校验的情况下,数据包中不包含校验位。
二、程序连接串口操作
SerialPort类的常用API:
属性:
常用方法
常用事件
代码示例:
using System.IO.Ports;
using System.Text;
namespace SerialPortCommunication
{
class Program
{
private static SerialPort serialPort;
public static void Main()
{
#region 串口通信
ConnectSerialPort("COM2");
ClosePort();
ReciveData();
SendData();
Console.ReadKey();
#endregion
}
/// <summary>
/// 接收串口数据
/// </summary>
/// <exception cref="NotImplementedException"></exception>
private static void ReciveData()
{
// 连接串口
var result = ConnectSerialPort("COM2");
while(true)
{
if (result)
{
// BytesToRead 代表串口中的字节大小
// 串口中的数据接收后,数据就被移除了。
var buffer = new byte[serialPort.BytesToRead];
// 接收发送的数据
#region 主动接收
serialPort.Read(buffer, 0, buffer.Length);
var resultString = Encoding.UTF8.GetString(buffer, 0, buffer.Length);
Console.WriteLine($"接收到结果: {resultString}");
}
}
serialPort.Close();
#endregion
}
/// <summary>
/// 发送数据的方法
/// </summary>
/// <exception cref="NotImplementedException"></exception>
private static void SendData()
{
// 连接串口
var result = ConnectSerialPort("COM1");
if (result)
{
// 发送数据
//serialPort.Write("Hello SerialPort");
//var buffer = new byte[] { 0x01, 0x02,0x03 };
// 发送数据
var buffer = Encoding.UTF8.GetBytes("中文");
serialPort.Write(buffer, 0, buffer.Length);
//serialPort.WriteBufferSize = 65535; // modbus 250
//serialPort.WriteLine("Don't type chinese");
}
serialPort.Close();
}
/// <summary>
/// 关闭串口的方法
/// </summary>
/// <exception cref="NotImplementedException"></exception>
private static void ClosePort()
{
if (serialPort == null)
{
return;
}
serialPort.Close();
}
/// <summary>
/// 连接串口的方法
/// </summary>
/// <returns></returns>
private static bool ConnectSerialPort(string portName)
{
// 创建串口对象
serialPort = new SerialPort();
// 设置连接串口名称
serialPort.PortName = portName;
// 设置波特率
serialPort.BaudRate = 9600;
// 设置数据位
serialPort.DataBits = 8;
// 设置停止位
serialPort.StopBits = StopBits.One;
// 设置校验位
serialPort.Parity = Parity.None;
serialPort.DataReceived += SerialPort_DataReceived;
// 打开串口
serialPort.Open();
// 判断串口是否成功打开
if (serialPort.IsOpen)
{
Console.WriteLine("串口连接成功");
return true;
}
else
{
Console.WriteLine("串口连接失败");
}
return false;
}
/// <summary>
/// 接收事件
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
/// <exception cref="NotImplementedException"></exception>
private static void SerialPort_DataReceived(object sender, SerialDataReceivedEventArgs e)
{
var serialPort1 = sender as SerialPort;
var buffer = new byte[serialPort1!.BytesToRead];
serialPort1.Read(buffer, 0, buffer.Length);
var resultString = Encoding.UTF8.GetString(buffer, 0, buffer.Length);
Console.WriteLine($"接收到结果: {resultString}");
}
}
}