接上面二.中文Win 95/98下的通信问题与解决方法 
  1.接收的数据少于发送的数据 
  如果通过MSComm控件一次性传送较多的二进制数据,那么,很 可能收到的数 
据不足 
。例如在设置为2400bps传输率的情况下,一次 性可以传输2048个字符数据,那么 
在大 
多数情况下一次只能收到 1200个字符左右,这是因为新版的MSComm32.OCX中存在 
一个影 
响传 输二进制数据的臭虫(bug),注意这不是特性。 
  32位Windows API函数(以下简称API)使用了几个用 COMMTIMEOUTS结构表示 
的限时 
变量,WriteTotalTimeOutConstant即 是其中的一个,它被Windows内部设定为 
5000(即 
5秒),这个常量 决定了在通信驱动程序停止传输之前花费在发送缓冲区中数据的 
时 间 
时 间 
的长短。5秒钟意味着通信速度为1200bps情况下仅能发送600个字 符,2400bps情 
况下仅 
能发送1200个左右的字符。事实上,在一个缓 冲区内一次性发送更多的数据是非 
常可能 
的。这个bug同样也能引发 问题,甚至在高速串口通信情况下,即使系统在使用流 
控制 
,无论 是软件流(Xon/Xoff)还是硬件流(CTS/RTS)。假如数据在发送缓冲区 中时 
流控制 
停止了传输,如果停止时间超过5秒钟,则数据就会丢 失。在某些环境下,5秒钟 
可能相 
当短,不过也不必担心,VB 5.0/6.0版本的MSComm控件有一个新增的重要的属性称 
为Co 
mmID, CommID指的是当串口被打开时,被API所调用的串口句柄或称标志, 这也 
意味着 
能利用API接口函数去修改这个常量。每次串口关闭后, Windows会自动将之恢复 
为500 
0,所以,每次打开串口后需要重新设 定以下API声明,其代码如下。 
  Type COMMTIMEOUTS 
  ReadIntervalTimeout As Long ReadTotalTimeoutMultiplier As Long 
ReadTotal 
  ReadIntervalTimeout As Long ReadTotalTimeoutMultiplier As Long 
ReadTotal 
ReadTotal 
TimeoutConstant As Long 
  WriteTotalTimeoutMultiplier As Long 
  WriteTotalTimeoutConstant As Long 
  End Type 
  Declare Function SetCommTimeouts Lib "Kernel32" (ByVal hFile As 
Long,lpC 
omm TimeoutsAs COMMTIMEOUTS) As Long 
  Declare Function GetCommTimeouts Lib "Kernel32" (ByVal hFile As 
Long,lpC 
ommTimeouts As COMMTIMEOUTS) As Long 
  Dim timeouts As COMMTIMEOUTS 
  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. 
Setti 
ngs))*10000+1000 
  Ret=SetCommTimeouts(Comm1.CommID,timeouts) 
  Ret=SetCommTimeouts(Comm1.CommID,timeouts) 
  (程序1) 
  2.如何发送大于128的字符数据 
  在通信程序中,以单字符方式逐个发送数据时,每一个数据范 围为0-255(即 
十六 
进制的00-FF)。在单字符版本的英文Win 95或 DOS版的BASIC程序中,只需要将相 
应的 
数据转换成相应的字符发送 到通信端口即可。但在中文Win 95/98下却行不通,假 
设在 
中文Win 95/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字符, 
大于1 
28的数字 仅作为前导字符,它只是显示是一个非拉丁语系的字符,而并不代 表实 
际意 
义。上述程序在调用CHR()函数时用到了DBCS字符集,因此 产生了此类错误。那么 
,如 
何发送大于128的数据呢?答案是使用字 节数组,将以上程序改为: 
  Dim cc(255) As Byte 
  For i = 0 To 255 
   cc(i) = i 
  Next i 
  MSComm1.Output = cc 
  Do 
  Do 
   DoEvents 
  Loop Until MSComm1.OutBufferCount = 0 
   
  '接收过程 MSComm1_OnComm() 
  Select Case MSComm1.CommEvent 
  Case comEvReceive 
   Dim Buffer As Variant,b1,i 
   MSComm1.InputMode = comInputModeBinary 
   MSComm1.InputLen = 0 
   Buffer = MSComm1.Input 
   For i=LBound(Buffer) To UBound(Buffer) 
   Debug.Print Buffer(i); 
   Next i 
  case ..... 
  3.如何发送0字符(00H,NULL) 
  在Visual C++中使用串口控件发送0字符有些麻烦,但在VB 5.0/6.0中只要注 
意以下 
两点即可: 
  (1)设置MSComm控件的属性NullDiscard=False; 
  (2)使用二进制接收,即用 MSComm1.InputMode=comInputModeBinary便可以解 
决问 
题; 
  4.如何发送中文字符串(DBCS字符) 
  VB 5.0/6.0的各种参考书上均指明MSComm通信控件不能发送或 接收双字节字 
符集系 
  VB 5.0/6.0的各种参考书上均指明MSComm通信控件不能发送或 接收双字节字 
符集系 
统(DBCS)的二进制数据,这对于我国及亚洲一 些使用DBCS字符集的国家不能不说 
是一大 
遗憾。但是我在实践中发 现,用MSComm控件也可以发送中文字符,具体方法有两 
种: 
种: 
  (1)直接发送 
  直接发送即把中文字符等同于英文字符。如: MSComm1.output="这是一行中 
文数据 
!",但这种方法发送的中文数 据不能太长,发送缓冲区和接收缓冲区的大小需设 
定为 
中文字符的 两倍以上,而且发送与接收系统所处的操作系统版本最好要一致, 否 
则会 
出现接收或发送缓冲区溢出之类的错误。这种方法可用于一 般要求不太高的场合 
。 
  (2)间接发送 
  在发送端将汉字或字符转换为机器内码或区位码数据数组,然 后将转换后的 
数据发 
送到串口,在接收端接收到数据后,按照相反 的顺序将得到的数据转换为相应的 
汉字或 
字符。在转换过程中,要 用到位运算,如取得汉字的内码后需要将高字节和低字 
节分开 
,而 VB 5.0/6.0中并没有提供此类函数,以下是求整数高、低字节的函 数。 
  Public Function HiByte(a As Integer) 
   Dim b 
   b = a And &HFF00 
   b = b / 256 
   If b < 0 Then b = b + 256 
   HiByte = b 
  End Function 
   
  Public Function LowByte(a As Integer) 
   Dim b 
   Dim b 
   b = a And &HFF 
   LowByte = b 
  End Function 
   
  5.如何更精准地计算时间差 
  在通信过程中,特别是在工业控制中,需要每隔若干秒做某一 件工作,即延 
时,以 
下是延时函数代码。 
  Public Sub Delay(PauseTime As Single) 
   Dim Start 
   Start = Timer 
   '设定开始时间 
   Do While Timer 
   DoEvents 
   Loop 
  End Sub 
  End Sub 
  实际上,当PauseTime<0.05秒时,Delay函数几乎不起作用,即 Timer函数几 
乎每次 
都得到相同的时间,只有大约隔了0.005秒才会 得到不同的时间,也就是说Timer 
的准确 
性只有0.05秒,但希望进行 的工作却是每0.005秒一次,该怎么办呢?可以改用 
Window 
s API的 TimeGetTime函数,该函数会传回Windows开机以来所经过的时间, 时间 
单位是 
1/1000秒,例如,开机经过2分钟,则传回值等于 2*60*1000。TimeGetTime的优点 
是可 
以将时间精确到1/1000秒,所 以可以用来解决上述问题,具体代码如下: 
  'API的声明 
  'API的声明 
  Declare Function timeGetTime Lib "winmm.dll" Alias "timeGetTime" 
() As L 
ong 
  '延时函数 
  Public Sub Delay(PauseTime As Single) 
   Dim Start 
   Start = timeGetTime 
   '设定开始时间 
   Do While Timer 
   DoEvents 
   Loop 
  End Sub 
  6.如何用单机进行通信测试 
  通常在写好了通信程序后需要两台PC或一台PC、一台单片机, 将通信口连接 
后进行 
测试,但很多时侯因条件限制仅有单台PC机, 测试项目很简单,那么能否测试呢 
?当然 
可以,而且方法也很简 单。对于九针的串口,找一个废弃的串口鼠标,剥开鼠标 
线,将 
连 接2、3针的线对接即可;对于25针的串口,可找一枚曲别针(最好 有塑料外套 
的) 
将它扯直,剥去两头的塑料后在两头各弯一个圆 圈,中间对折后直接套接在串口 
的2、 
3针上即可。如果担心不够安 全,则可以将5针接地。 
  以上问题均经过测试,并且已经成功地应用于本人开发的远程 字幕处理系统 
中,这 
在各类通信问题中也具有一定的普遍性。