VB下对串行接口第9位的操作以及API实现方法

2023-10-11 13:32

本文主要是介绍VB下对串行接口第9位的操作以及API实现方法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

RS232-485串行接口是一种非常成熟的通信接口,曾几何时,我们用的鼠标是串口的,Modem是串口的,还有早期的一些数码相机都是串口的,时过境迁,家用电脑现在已是USB时代,串口这种东西逐渐淡出了我们的视线。 但是,在工业控制上,串行接口依然有着不可替代的优势,首先是电气连接简单,虽说速率不高,但抗干扰能力强,通讯距离很远,甚至可以铺设几百米的电缆,这些都是USB不能取代的。

对串行接口的操作,微软公司很早前就提供了一个通用控件,她就是大名鼎鼎的MSCOMM,这个控件可以嵌入几乎所有宿主语言,包括主流的VC VB DELPHI C++ Build等等。通过这个控件,我们可以极其轻易地对串口进行操作。但是,这个控件依然不是完美的,因为微软在写这个控件的时候,考虑的都是一般性的常规的操作,不过,一旦遇到非常规操作,控件立刻就显示出它的局限性,正如可视化编程下控件滥用的坏习惯一样,没有人再去花心思研究程序的内部原理,鼠标拖一下,键盘敲几个调用,甚至一个程序就出来了,这并不是好事,一旦遇到非常规事务,立刻就会束手无策。

把话题拉回串行接口,串型数据RS232接口的基本概念是以高低脉冲来区分0或者1,以一个字节(Byte)为最小单位进行发送,一个Byte为8个二进制位(BIT),另外附加三个位作为起始位、停止位和奇偶校验位。在选择不使用奇偶校验的情况下,串口一次最小传送10个BIT,如果需要奇偶校验,则是11个BIT,排列如下:

[起始位] [数据位1到8] [奇偶校验位] [停止位]

奇偶校验的原理是,计算数据位内上升沿的个数,也就是BIT=1的次数,然后再根据这个个数据决定奇偶校验位是0还是1,比如说发送1这个数,并且现在我们选用奇校验,则奇偶校验位是0,因为(原始数据=1,奇偶校验位=0,1+0=1),1是奇数。 如果选用偶校验,则奇偶校验位会自动变成1,(原始数据=1,奇偶校验位=1,1+1=2),2是偶数。发送方将数据和奇偶校验位一起发送,接受方开始接收数据,并且核对奇偶校验位,一旦发现奇偶校验位有误,则立刻报错,因为这说明数据传输受到了干扰。

奇偶校验位一般被称为串口的“第九位”,这个位其实除了校验数据外,还有别的另类玩法。在主机上利用串行接口对多设备进行控制的时候,主机发送到每一条命令,必须要编上地址才行,否则就变成广播操作了,就像老大一声吼,底下的小弟们全部振臂狂呼,这在某些时候确实有用,但如果老大只点了一个小弟的名字让他单独回答,就会出问题了,人类于是有了名字,而在工业控制上,模块都需要编上地址,这跟名字其实没什么本质上的区别。串行数据流里面,往往利用第九位来区分是地址包还是数据包,大家约定,凡是第九位为1的BYTE,说明这是地址,凡是第九位为0的BYTE,那是数据。主机控制下的各分机只有在接受到第九位为1的时候,才进行地址识别,如果确实与主机呼叫的地址一致,才开始识别接下来的数据(第九位为0)。可以看出,这样的方式是很聪明的,各分机没有必要频繁地接收主机发送到数据流,只有收到第九位为1并且符合自己地址之后,才进行接收,效率不言而喻。

如果采用第九位作为地址/数据的区分,那么串口将丧失奇偶校验功能,这是没有办法的事,鱼与熊掌不可兼得嘛。所以在Windows串行接口规范里,对这个位有5种设置,分别是:
NOPARITY = 无校验
ODDPARITY = 偶校验
EVENPARITY = 奇校验
MARKPARITY = 第九位强设为1
SPACEPARITY = 第九位强设为0

在发地址包的时候, 可以把Parity设置成MARKPARITY. 则第九位常为1.
在发数据包的时候, 可以把Parity设置成SPACEPARITY.则第九位常为0.

看起来不困难,无非就是改变第九位的状态而已嘛。但是,很快,可怕的事情来了,使用MSCOMM控件的话,如果频繁地改动奇偶校验操作,则通讯将会出现丢包等莫名其妙的问题!但我们为了区分数据和地址,这种频繁改动又是必须的,怎么办?只能扔掉MSCOMM,另寻他途了。

利用API搭建一个串口通讯程序,是一个好办法,API程序直接作用于Windows,效率很高,VC++用的类库MFC无非也就是将成千上万的API函数集中起来并加以聚合,抽象。现在我们直接使用API,当然是可行的,但是,因为Visual Basic本身的缺陷,她没办法像VC那样创建多线程程序(至少实现起来极其困难),在以下的例子里我们只能采用同步的方法来获得串口的数据而不能实现异步接收,等等,到底什么叫同步?异步?简单地说,比如你拖一个1G的文件从C盘到D盘,这需要大量的时间,如果这段时间系统一直等着它完成COPY的操作,其他什么都不管理,那么这就叫同步(回忆一下DOS时代不就是这样的吗)。但是,如果系统只是给它这么一条指令,然后你该什么时候COPY完后通知我一声,让我知道你COPY完了就行了,系统在这段时间内不会死等这个操作完成,而是释放开给别的有需要的程序(在Windows时代,你可以边COPY边听歌),这就叫异步。很显然,异步操作聪明得多,也比较合理,最大的优势是榨干了CPU的效能,但鉴于VB这方面完全不行,所以也只好采用同步的方法了。

以下是源代码:
API声明:

Option Explicit


'奇偶校验常数
Public Const NOPARITY = 0
Public Const ODDPARITY = 1
Public Const EVENPARITY = 2
Public Const MARKPARITY = 3
Public Const SPACEPARITY = 4
'-------------------------------------------------------------------------------
' 文件操作常数
'-------------------------------------------------------------------------------
Public Const ERROR_IO_INCOMPLETE = 996&
Public Const ERROR_IO_PENDING = 997
Public Const GENERIC_READ = &H80000000
Public Const GENERIC_WRITE = &H40000000
Public Const FILE_ATTRIBUTE_NORMAL = &H80
Public Const FILE_FLAG_OVERLAPPED = &H40000000
Public Const FORMAT_MESSAGE_FROM_SYSTEM = &H1000
Public Const OPEN_EXISTING = 3

' 通讯常数
Public Const MS_CTS_ON = &H10&
Public Const MS_DSR_ON = &H20&
Public Const MS_RING_ON = &H40&
Public Const MS_RLSD_ON = &H80&
Public Const PURGE_RXABORT = &H2
Public Const PURGE_RXCLEAR = &H8
Public Const PURGE_TXABORT = &H1
Public Const PURGE_TXCLEAR = &H4

'-------------------------------------------------------------------------------
'通讯结构
'-------------------------------------------------------------------------------
Public Type COMSTAT
fBitFields As Long ' See Comment in Win32API.Txt
cbInQue As Long
cbOutQue As Long
End Type

Public Type COMMTIMEOUTS
ReadIntervalTimeout As Long
ReadTotalTimeoutMultiplier As Long
ReadTotalTimeoutConstant As Long
WriteTotalTimeoutMultiplier As Long
WriteTotalTimeoutConstant As Long
End Type

'
'DCB结构,用于串口的设置

Public Type DCB
DCBlength As Long
BaudRate As Long
fBitFields As Long
wReserved As Integer
XonLim As Integer
XoffLim As Integer
ByteSize As Byte
Parity As Byte
StopBits As Byte
XonChar As Byte
XoffChar As Byte
ErrorChar As Byte
EofChar As Byte
EvtChar As Byte
wReserved1 As Integer 'Reserved; Do Not Use
End Type

'各种API函数的声明:
'建立通讯连接
Public Declare Function CreateFile Lib "kernel32" Alias "CreateFileA" (ByVal lpFileName As String, ByVal dwDesiredAccess As Long, ByVal dwShareMode As Long, ByVal lpSecurityAttributes As Long, ByVal dwCreationDisposition As Long, ByVal dwFlagsAndAttributes As Long, ByVal hTemplateFile As Long) As Long
'关闭通讯连接
Public Declare Function CloseHandle Lib "kernel32" (ByVal h Object As Long) As Long
'发送数据
Public Declare Function WriteFile Lib "kernel32" (ByVal hFile As Long, lpBuffer As Any, ByVal nNumberOfBytesToWrite As Long, lpNumberOfBytesWritten As Long, lpOverlapped As Long) As Long
'读取数据
Public Declare Function ReadFile Lib "kernel32" (ByVal hFile As Long, lpBuffer As Any, ByVal nNumberOfBytesToRead As Long, lpNumberOfBytesRead As Long, ByVal lpOverlapped As Long) As Long
'获取DCB串口设置状态
Public Declare Function GetCommState Lib "kernel32" (ByVal nCid As Long, lpDCB As DCB) As Long
'构建DCB串口设置状态
Public Declare Function BuildCommDCB Lib "kernel32" Alias "BuildCommDCBA" (ByVal lpDef As String, lpDCB As DCB) As Long
'设置DCB串口设置状态
Public Declare Function SetCommState Lib "kernel32" (ByVal hCommDev As Long, lpDCB As DCB) As Long
'设置串口的缓冲区
Public Declare Function SetupComm Lib "kernel32" (ByVal hFile As Long, ByVal dwInQueue As Long, ByVal dwOutQueue As Long) As Long
'清除串口缓冲区的数据
Public Declare Function PurgeComm Lib "kernel32" (ByVal hFile As Long, ByVal dwFlags As Long) As Long
'设置串口的超时状态
Public Declare Function SetCommTimeouts Lib "kernel32" (ByVal hFile As Long, lpCommTimeouts As COMMTIMEOUTS) As Long
'获取错误状态
Public Declare Function GetLastError Lib "kernel32" () As Long
'产生一个系统延时,单位毫秒
Public Declare Sub Sleep Lib "kernel32" (ByVal dwMilliseconds As Long)

============================================================

以下是程序代码:

'全局变量hCF为通讯句柄
Dim hCF As Long

Private Sub Form_Load()
'建立通讯连接
hCF = CreateFile("COM1", _
GENERIC_READ Or GENERIC_WRITE, 0, ByVal 0&, _
OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0)

End Sub


Private Sub Command1_click()
Dim Ret As Long
Dim Buffer(30) As Byte
Dim I As Long

Dim typCommStat As COMSTAT '定义串口状态结构
Dim lngError As Long '定义串口状态错误
Dim flag As Long '定义回传值
Dim typDCB As DCB '定义DCB串口设置块
Dim strSettings As String

flag = SetupComm(hCF, 1024, 1024) '设置缓冲区大小,1K

'强制清空读写缓冲区
flag = PurgeComm(hCF, PURGE_RXABORT Or PURGE_RXCLEAR Or PURGE_TXABORT Or PURGE_TXCLEAR)

'定义超时结构体
Dim typCommTimeouts As COMMTIMEOUTS
typCommTimeouts.ReadIntervalTimeout = 0 '相邻两字节读取最大时间间隔(为0表示不使用该超时间隔)
typCommTimeouts.ReadTotalTimeoutMultiplier = 10 '一个读操作的时间常数
typCommTimeouts.ReadTotalTimeoutConstant = 10 '读超时常数
typCommTimeouts.WriteTotalTimeoutMultiplier = 0 '一个写操作的时间常数(为0表示不使用该超时间隔)
typCommTimeouts.WriteTotalTimeoutConstant = 0 '写超时常数(为0表示不使用该超时间隔)
'超时设置
flag = SetCommTimeouts(hCF, typCommTimeouts)


Dim addressByte(0 To 1) As Byte '地址位,两个字节
Dim dataByte(0 To 3) As Byte '数据位,四个字节

flag = GetCommState(hCF, typDCB)
strSettings = "baud=19200 parity=m data=8 stop=1" '首先将奇偶校验位调节到M模式,则强制设为1
flag = BuildCommDCB(strSettings, typDCB) '构建DCB块
flag = SetCommState(hCF, typDCB) '设置DCB块

addressByte(0) = &H0 '分机编号0000,占用两个字节
addressByte(1) = &H0
Ret = WriteFile(hCF, addressByte(0), 2, flag, ByVal 0&) '发送

flag = GetCommState(hCF, typDCB)
strSettings = "baud=19200 parity=s data=8 stop=1" '首先将奇偶校验位调节到S模式,则强制设为0
flag = BuildCommDCB(strSettings, typDCB)
flag = SetCommState(hCF, typDCB)
flag = GetCommState(hCF, typDCB)

dataByte(0) = &H3 '这是数据,我的数据为4个字节,这个依据实际情况自行定义
dataByte(1) = &H20
dataByte(2) = &H0
dataByte(3) = &H23
Ret = WriteFile(hCF, dataByte(0), 4, flag, ByVal 0&) '发送

Sleep 50 '延时50毫秒

'同步接收来自串口的数据,数据存到Buffer数组里,我这里取30字节,这个可以按实际情况自定
Ret = ReadFile(hCF, Buffer(0), 30, 0, 0)

For I = 0 To 30
Debug.Print Hex(Buffer(I)) '在DEBUG窗口显示接收过来的数据
Next I

End Sub


Private Sub Form_Unload(Cancel As Integer)
CloseHandle hCF '关闭通讯连接
End Sub
 

这篇关于VB下对串行接口第9位的操作以及API实现方法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/188397

相关文章

Linux使用fdisk进行磁盘的相关操作

《Linux使用fdisk进行磁盘的相关操作》fdisk命令是Linux中用于管理磁盘分区的强大文本实用程序,这篇文章主要为大家详细介绍了如何使用fdisk进行磁盘的相关操作,需要的可以了解下... 目录简介基本语法示例用法列出所有分区查看指定磁盘的区分管理指定的磁盘进入交互式模式创建一个新的分区删除一个存

Window Server2016加入AD域的方法步骤

《WindowServer2016加入AD域的方法步骤》:本文主要介绍WindowServer2016加入AD域的方法步骤,包括配置DNS、检测ping通、更改计算机域、输入账号密码、重启服务... 目录一、 准备条件二、配置ServerB加入ServerA的AD域(test.ly)三、查看加入AD域后的变

windos server2022里的DFS配置的实现

《windosserver2022里的DFS配置的实现》DFS是WindowsServer操作系统提供的一种功能,用于在多台服务器上集中管理共享文件夹和文件的分布式存储解决方案,本文就来介绍一下wi... 目录什么是DFS?优势:应用场景:DFS配置步骤什么是DFS?DFS指的是分布式文件系统(Distr

Golang操作DuckDB实战案例分享

《Golang操作DuckDB实战案例分享》DuckDB是一个嵌入式SQL数据库引擎,它与众所周知的SQLite非常相似,但它是为olap风格的工作负载设计的,DuckDB支持各种数据类型和SQL特性... 目录DuckDB的主要优点环境准备初始化表和数据查询单行或多行错误处理和事务完整代码最后总结Duck

Window Server2016 AD域的创建的方法步骤

《WindowServer2016AD域的创建的方法步骤》本文主要介绍了WindowServer2016AD域的创建的方法步骤,文中通过图文介绍的非常详细,对大家的学习或者工作具有一定的参考学习价... 目录一、准备条件二、在ServerA服务器中常见AD域管理器:三、创建AD域,域地址为“test.ly”

NFS实现多服务器文件的共享的方法步骤

《NFS实现多服务器文件的共享的方法步骤》NFS允许网络中的计算机之间共享资源,客户端可以透明地读写远端NFS服务器上的文件,本文就来介绍一下NFS实现多服务器文件的共享的方法步骤,感兴趣的可以了解一... 目录一、简介二、部署1、准备1、服务端和客户端:安装nfs-utils2、服务端:创建共享目录3、服

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

C#使用yield关键字实现提升迭代性能与效率

《C#使用yield关键字实现提升迭代性能与效率》yield关键字在C#中简化了数据迭代的方式,实现了按需生成数据,自动维护迭代状态,本文主要来聊聊如何使用yield关键字实现提升迭代性能与效率,感兴... 目录前言传统迭代和yield迭代方式对比yield延迟加载按需获取数据yield break显式示迭

Python实现高效地读写大型文件

《Python实现高效地读写大型文件》Python如何读写的是大型文件,有没有什么方法来提高效率呢,这篇文章就来和大家聊聊如何在Python中高效地读写大型文件,需要的可以了解下... 目录一、逐行读取大型文件二、分块读取大型文件三、使用 mmap 模块进行内存映射文件操作(适用于大文件)四、使用 pand

python实现pdf转word和excel的示例代码

《python实现pdf转word和excel的示例代码》本文主要介绍了python实现pdf转word和excel的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价... 目录一、引言二、python编程1,PDF转Word2,PDF转Excel三、前端页面效果展示总结一