C C++最全VB和51单片机串口通信讲解(只针对VB部分)_vb与单片机串口通信,2024年最新C C++面试你必须要知道的那些知识

img
img

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

作为一个vB程序员,要编写通信程序.只需知道通信控件提供给Windows通信AP1函数的接口即可.换句话说,只需设定和监视通

信控件的属性和事件即可。

2.使用Mscomm控件

在开始使用MSComm控件之前。需要先了解其属性、事件或错误

属性            描述

CommPort    设置或返回通信端口号

Settings    以字符串的形式设置或返回波特率、奇偶校验、数据位和停止位

PortOpen    设置或返回通信端口的状态。也可以打开和关闭端口

Input       返回和删除接收缓冲区中的字符

Output      将字符串写入发送缓冲区

CommEvent属性为通信事件或错误返回下列值之一。在该控件的对象库中也可以找到这些常量。

常量           值        描述

ComEventBreak   1001    收到了断开信号

ComEventCTSTO   1002    Clear To Send Timeout。在发送字符时,在系统指定的事1件内,CTS(Clear To Send)线是低电平

ComEventDSRTO   1003    Data Set Ready Timeout。在发送字符时,在系统指定的事件内,DSR(Data Set Ready)线是低电平

ComEventFrame   1004    数据帧错误。硬件检测到一个数据帧错误

ComEventOverrun 1006    端口溢出。硬件中的字符尚未读,下一个字符又到达,并且丢失

ComEventCDTO    1007    Carrier Detect Time。在发送字符时,在系统指定的事件内,CD(Carrier Detect)线是低电平。CD

也称为RLSD(Receive Line Singal Detect,接收线信号检测)

ComEventRxOver 1008    接收缓冲区溢出。在接收缓冲区中没有空间

ComEventRxParity 1009   奇偶校验错。硬件检测到奇偶校验错误7

ComEventTxFull 1010    发送缓冲区满。在对发送字符排队时,发送缓冲区满

ComEventDCB     1011    检取端口DCB(Device Control Blick)时发生了没有预料到的错误

通信事件包含了下面的设置:

常量         值        描述

ComEvSend      1    发送缓冲区中的字符数比Sthreshold值低

ComEvReceive   2    接收到了Rthreshold个字符。持续产生该事件,直到使用了Input属性删除了接收缓冲区中的数据

ComEvCTS       3    CTS(Clear To Send)线改变

ComEvDSR       4    DSR(Data Set Ready)线改变。当DSR从1到0改变时,该事件发生

ComEvCD        5    CD(Carrier Detect)线改变ComEvRing6检测到响铃信号。一些URAT(Universal AsynchronousReciver-

-Transmitters,通用异步收发器)不支持该事件

ComEvEOF       7    收到了EOF字符(ASCII字符26)

Error消息(MSComm控件)下表列出了MSComm控件可捕获的错误消息:

常量                       值      描述

ComInvalidPropertyValue    380   无效的属性值

ComSetNotSupported         383   属性只读

ComGetNotSupported         394   属性只读

ComPortOpen               8000   端口打开时该存在无效

8001   超时设置必须比0值大

ComPortInvalid            8002   无效的端口号

8003   属性只在运行时有效

8004   属性在运行时是只读的

ComPortAleadyOpen         8005   端口已经打开

8006   设备标识符无效或不支持

8007   不支持设备的波特率

8008   指定的字节大小无效

8009   缺省参数错误

8010   硬件不可用(被其他设备锁住)

8011   函数不能分配队列

ComNoOpen                 8012   设备没有打开

8013   设备已经打开

8014   不能使用通信通知

ComSetCommStateFailed     8015   不能设置通信状态

8016   不能设置通信事件屏蔽

ComPortNotOpen            8018   该存在只在端口打开是有效

8019   设备忙

ComReadError              8020   通信设备读错误

ComDCBError               8021   检取端口设备控制块时出现内部错误

搞清楚以上基本属性后,就可以开始编写通信许程序了。在VB5.0/6.0中新建一个工程文件。添加Microsoft Comm Control 5.0组

件,在简体Form1中加入Command命令按钮并取名为CmdTest,MSComm控件取名为MSComm1,加入如下程序代码。

Private Sub cmdTestClick ( )            '打开串口

MSComml.CommPort =2                     '设定Com2

If MSComml.PortOpen = False Then

MSComm1.Settings = “9600,n,8,1”         '9600波特率,无校验,8位数据位,1位停止位

MSComm1.PortOpen = True                 '打开串口

End if

MSComm1.OutBufferCount = 0              '清空发送缓冲区

MSComm1.InBufferCount = 0               '滑空接收缓冲区

'发送字符数据时注意必须用回车符(vbcr)结束

MSComm1.Output="This is a qood book ! " &vbCr

'泼打电话号码或发送AT命令

MSComm1.Output = "ATDT 05778191898 , & vbCr

'发送字符数组数据时注意ByteArray必须事先定义赋值

Dim ByteArray as byte( )

'定义动态数组

ReDim ByteArray(1)

'重定义数组大小

ByteArray ( 0 ) =0

ByteArray ( 1 ) = 1

MSComm1.Output = ByteArray

End Sub

private Sub MScommEvent( )

Select Case MSComm1.CommEvent

Case comEvReceive

Dim Buffer As Variant

MSComm1.InputLen = 0

'接收二进制数据

MSComm1.InputMode= ComInputModeBinary

Buffer=MSComm1.Input

'接收字符数据

MSComm1.InputMode=comInputModeText

Buffer = MSComml.Input

Case else

End Select

End sub

( 程序1)


二、中文Win 95/98下的通信问题与解决方法
1.接收的数据少于发送的数据
     如果通过MSComm控件一次性传送较多的二进制数据,那么,很可能收到的数据不足。例如在设置为24oobps传输率的情况下,
一次性可以传输2048个字符数据那么在大多数情况下。一次只能收到1200个字符左右,这址出为新版的MSComm32.OCX中存在一
个影响传输二进制数据的臭虫(bug).注意这不是特性。
     32位Windows API函数(以下简称API)使用了几个用COMMTIMEOUTS结构表示的限时变量,WriteTotalTimeOutConstant 即是其
中的一个,它被Windows内部设定为5000(即5秒),这个常量决定了在通信驱动程序停止传输之前花费在发送缓冲区中数据的时间
的长短,5秒钟意味着通信速度为1200bps情况下仅能发送600个字符,24oobps情况下仅能发送1200个左右的字符。事实上,在一个
缓冲区内一次性发送更多的数据是非常可能的。这个bug同样也能引发问题,甚至在高速串口门通信情况下,即使系统在使用流控
制,无论丛软件流(Xon/XofI)还是硬件流(CTS/RTS)。假如数据在发送缓冲区中时,流控制停止了传输,如果停止时间超过5
秒钟.则数据就会丢失。在某些环境下,5秒钟可能相当短.不过也不必担心, VB 5.0/6.0版本的MSComm控件有一个新增的重要的
属性称为CommID, CommID指的是当串口被打开时,被API所调用的串口句柄或称标志,这也意味着能利用API接口函数去修改这个
常量。每次串口关闭后,Windows会自动将之恢复为5000,所以,每次打开串口后需要重斩设定以下API声明,其代码见下程序。
Type COMMTIMEOUTS
ReadIntervalTimeout As Long
ReadTotalTimeoutMultiplier As Long
ReadTotalTimeoutConstant As Long
WriteTotalTimeoutMultiplier As Long
WriteTotalTimeoutConstant As Long
End Type
Declare Function SetCommTimeouts Lib “Kernel32”
(BYVal hFile As Long, lpComm TimeoutsAs COMMTIMEOUTS) As Long
Declare Function GetCommTimeouts Lib “Kernel32”
(ByVal hFile As Long, lpCommTimeouts As COMMTIMEOUTS) As Long
Dim timeouts As COMMEOUTS
Dim Ret As Long
If Comm1.PortOpen = False Then
Comm1.PortOpen = True
End if
Ret=GetCommTimeouts ( Comm1.CommID , timeouts )
'Set some default timeouts
timeOuts.ReadIntervalTimeout = 1
timeouts.ReadTotalTimeoutMultiplier =1
timeouts.ReadTotalTimeoutConstant =1
timeouts.WriteTotalTimeoutMultiplier =1
timeouts.WriteTotalTimeoutConstant=
( Comm1.OutBufferSize\Val(Comm1.Settings))*10000+1000
Ret=SetCommTimeouts( Comm1.CommID , timeouts )
( 程序2)

2.如何发送大于128的字符数据
     在通信程序中,以单字符方式逐个发送数据时,每一个数据范围 0-255(即十六进制的00-FF)。在单字符版本的英文Win95或
DOS版的BASIC程序中,只需要将相应的数据转换成相应的字符发送到通信端口即可。但在中文Win95/98下却行不通,假设在中文
Win95/98下运行以下程序:
Dim i
For i=0 to 255
MSComm1.Output=chr(i)
Next i

希望在接收端得到预期的0-255之间的数据,结果却是:前129个数据接收正确,为0-128,后面127个数据为126个0和一个255,
造成这种给果的原因在于中文Windows使用的是双字节字符集(DBCS)系统。DBCS系统使用0-128之间的数字表示ASCII字符,大于
128的数字仅作为前导字符,它只是显示是一个非拉丁语系的字符,而并不代表实际意义。上述程序在调用CHR()函数时用到了
DBCS字符集,冈此产生了此类错误。那么,如何发送人于128的数据呢?答案是使用字符数组,将以上程序改为:
Dim cc(255) As Byte
For i = 0 To 255
cc(i) = i
Next i
MSComm1.Output = cc
Do
DoEvents
Loop Until MSComm1.OutBufferCount = 0
'接收过程 MSComm1_OnComm()
Select Case MSComm1.CommEvent
Case comEvReceive
Dim Buffer As Variant, b1,i
MSComm1.InputMode=comInputModeBinery
MSComm1.InputLen = 0
Buffer = MSComm1.Input
For i=LBound (Buffer) To UBound (Buffer )
Debug.Print Buffer ( i ) ;
Next i
Case . . . . .

3.如何发送0字符(00H,NULL)
在VisuaI C++中使用串口控件发送0字符有些麻烦,但在VB5.0/6.0中只要注意以下两点即可:
(1)设置MSComm控件的属性 NullDiscard=False;。
(2)使用二进制接收,即用 MSComm1.InputMode=ComInputModeBinary便可以解决问题;

4.如何发送递中文字符串(DBcS字符)
VB5.0/6.0的各种参考书上均指明MSComm通信控件不能发送或接收双字节字符集系统DBCS)的二进制数据,这对于我国及亚洲一些
使用DBCS字符集的国家不能不说是一大人遗憾。但是我在实践中发现,用MSComm控件也可以发送中文字符,具体方法有以下两种:
(1)直接发送
     直接发送即把中文字符等同于英文字符。如:MSComm1.Intput= " 这是一行中文数据!" ,但这种方法发送的中文数据不能太
长,发送缓冲区和接收缓冲区的大小需设定为中文字符的两倍以上,而且发送与接收系统所处的操作系统版本最好要一致,否则会
出现接收或发送缓冲区溢出之类的错误。这种方法时用于一般要求不太高的场合。

img
img

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

的大小需设定为中文字符的两倍以上,而且发送与接收系统所处的操作系统版本最好要一致,否则会
出现接收或发送缓冲区溢出之类的错误。这种方法时用于一般要求不太高的场合。

[外链图片转存中…(img-37ChBh1V-1715698086137)]
[外链图片转存中…(img-lmdrOSUq-1715698086137)]

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 17
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值