📄 vc实现串口通信例程.html
字号:
</table></td>
</tr>
<tr>
<td height="3" bgcolor="#CCCCCC"></td>
</tr>
</table><table width="760" border="0" cellpadding="0" cellspacing="0" class="top_tdbgall" style="word-break:break-all;Width:fixed">
<tr>
<td></td>
</tr>
<tr>
<td align="center"><object classid='clsid:D27CDB6E-AE6D-11cf-96B8-444553540000' codebase='http://download.macromedia.com/pub/shockwave/cabs/flash/swflash.cab#version=5,0,0,0' width='755' height='120'><param name='movie' value='/img/61ban_0423.swf'><param name='wmode' value='transparent'><param name='quality' value='high'><embed src='/img/61ban_0423.swf' pluginspage='http://www.macromedia.com/shockwave/download/index.cgi?P1_Prod_Version=ShockwaveFlash' type='application/x-shockwave-flash' width='755' height='120'></embed></object></td>
</tr>
<tr align="center">
<td class="top_nav_menu"><table width="100%" border="0" align="center" cellpadding="0" cellspacing="0" class="top_nav_menu" style="word-break:break-all;Width:fixed">
<tr>
<td height="1" align="center" bgcolor="#999999"></td>
</tr>
<tr bgcolor="#FFFFFF">
<td height="2" align="center"></td>
</tr>
<tr>
<td height="22" align="right" bgcolor="eeeeee"><script language='JavaScript1.2' type='text/JavaScript' src='/Article/JS/ShowClass_Menu.js'></script></td>
</tr>
<tr bgcolor="#CCCCCC">
<td height="3" align="center"></td>
</tr>
</table></td>
</tr>
</table>
<!--导航代码开始-->
<table width="760" border="0" cellpadding="0" cellspacing="0" class="top_Path" style="word-break:break-all;Width:fixed">
<tr>
<td width="4%" align="center"><img src="/skin/adv3/tp009.gif" width="15" height="15"></td>
<td width="75%">您现在的位置: <a href='http://www.mcuchina.com'>中国单片机在线网</a> >> <a href='/Article/Index.html'>文章中心</a> >> <a href='/Article/Interface/Index.html'>接口技术</a> >> <a href='/Article/Interface/COM/Index.html'>串口技术</a> >> <a href='/Article/Interface/COM/COMPRG/Index.html'>串口编程</a> >> 正文</td>
<td width="21%"><img src="/Skin/adv3/User_Login.gif" align="absmiddle"> <a href="/User/User_Login.asp" target="_blank" class="top_UserLogin">用户登录</a> <img src="/Skin/adv3/User_Reg.gif" align="absmiddle"> <a href="/Reg/User_Reg.asp" target="_blank" class="top_UserLogin">新用户注册</a> </td>
</tr>
</table>
<!--导航代码结束-->
<!-- ********网页顶部代码结束******** -->
<!-- ********网页中部代码开始******** -->
<!--文章显示代码开始-->
<table width="760" border="0" cellpadding="0" cellspacing="0" class="center_tdbgall" style="word-break:break-all;Width:fixed">
<tr>
<td colspan="2"> <table width="100%" border="0" cellspacing="0" cellpadding="0" style="word-break:break-all;Width:fixed">
<tr>
<td width="8%" height="34" valign="middle" Class="main_title_575"><img src="/skin/adv3/homedha3.gif" width="55" height="34"></td>
<td width="53%" Class="main_title_575">VC实现串口通信例程</td>
<td width="20%" Class="main_title_575"> <font color='#009900'>★★★</font></td>
<td width="19%" Class="main_title_575" align='right'>【字体:<a href="javascript:fontZoomA();" class="top_UserLogin">小</a>
<a href="javascript:fontZoomB();" class="top_UserLogin">大</a>】</td>
</tr>
</table></td>
</tr>
<tr align="center" valign="middle">
<td height="50" colspan="2" style="word-break:break-all;Width:fixed">VC实现串口通信例程</td>
</tr>
<tr align="center" valign="middle">
<td colspan="2" style="word-break:break-all;Width:fixed"><span class="main_ArticleSubheading"></span></td>
</tr>
<tr align="center" bgcolor="#CCCCCC">
<td height="1" colspan="2"></td>
</tr>
<tr align="center">
<td height="2" colspan="2"></td>
</tr>
<tr align="center" class="left_tdbgall">
<td height="26" colspan="2">作者:阮帮秋 文章来源:<a href='http://www.gjwtech.com'>龚建伟技术主页</a> 点击数:<script language='javascript' src='/Article/GetHits.asp?ArticleID=57'></script> 更新时间:2004-11-24</td>
</tr>
<tr>
<td height="1" colspan="2" bgcolor="#CCCCCC"></td>
</tr>
<tr>
<td height="300" colspan="2" valign="top" class="main_tdbg_760" id="fontzoom" style="word-break:break-all;Width:fixed">
<table border="0" align="left" cellpadding="0" cellspacing="0">
<tr>
<td></td>
</tr>
</table>
<P align=left><FONT color=#386490><B class=font1>摘要:</B><SPAN class=line>WIN95界面下的VC++串口通讯程序在WIN32下是不建议对端口进行操作的,在WIN32中所有的设备都被看成是文件,串行口也不例外也是作为文件来进行处理的。<B class=font1><BR> 关键词</B></SPAN><B class=font1> </B>串行口,DWORD,缓冲区<B class=font1><!-- #EndEditable --> </B></FONT></P>
<P align=left><FONT color=#386490><BR><SPAN class=line> WIN95界面下的VC++串口通讯程序在WIN32下是不建议对端口进行操作的,在WIN32中所有的设备都被看成是文件,串行口也不例外也是作为文件来进行处理的。这是我的一份关于串口编程的读书笔记,对于使 用VC进行编程的同行应该有一定的帮助。</SPAN></FONT></P>
<P class=line><FONT color=#386490><B>1.打开串口:</B><BR><BR> 在Window 95下串行口作为文件处理,使用文件操作对串行口进行处理。使用CreateFile()打开串口,CreateFile()将返回串口的句柄。<BR> HANDLE CreateFile(<BR> LPCTSTR lpFileName, // pointer to name of the file<BR> DWORD dwDesiredAccess, // access (read-write) mode<BR> DWORD dwShareMode, // share mode<BR> LPSECURITY_ATTRIBUTES lpSecurityAttributes, // pointer to security attributes<BR> DWORD dwCreationDistribution, // how to create<BR> DWORD dwFlagsAndAttributes, // file attributes<BR> HANDLE hTemplateFile // handle to file with attributes to copy<BR> );<BR> lpFileName: 指明串口制备,例:COM1,COM2<BR> dwDesiredAccess: 指明串口存取方式,例:GENERIC_READ|GENERIC_WRITE<BR> dwShareMode: 指明串口共享方式<BR> lpSecurityAttributes: 指明串口的安全属性结构,NULL为缺省安全属性<BR> dwCreateionDistribution: 必须为OPEN_EXISTIN<BR> dwFlagAndAttributes: 对串口唯一有意义的是FILE_FLAG_OVERLAPPED<BR> hTemplatefile: 必须为NULL<BR><BR><B>2.关闭串口:</B><BR><BR> CloseHandle(hCommDev);<BR><BR><B>3.设置缓冲区长度:</B><BR><BR> BOOL SetupComm(<BR> HANDLE hFile, // handle of communications device<BR> DWORD dwInQueue, // size of input buffer<BR> DWORD dwOutQueue // size of output buffer<BR> );<BR><BR><B>4.COMMPROP结构:</B><BR><BR> 可使用GetCommProperties()取得COMMPROP结构,COMMPROP结构中记载了系统支持的各项设置。<BR> typedef struct _COMMPROP { // cmmp<BR> WORD wPacketLength; // packet size, in bytes<BR> WORD wPacketVersion; // packet version<BR> DWORD dwServiceMask; // services implemented<BR> DWORD dwReserved1; // reserved<BR> DWORD dwMaxTxQueue; // max Tx bufsize, in bytes<BR> DWORD dwMaxRxQueue; // max Rx bufsize, in bytes<BR> DWORD dwMaxBaud; // max baud rate, in bps<BR> DWORD dwProvSubType; // specific provider type<BR> DWORD dwProvCapabilities; // capabilities supported<BR> DWORD dwSettableParams; // changeable parameters<BR> DWORD dwSettableBaud; // allowable baud rates<BR> WORD wSettableData; // allowable byte sizes<BR> WORD wSettableStopParity; // stop bits/parity allowed<BR> DWORD dwCurrentTxQueue; // Tx buffer size, in bytes<BR> DWORD dwCurrentRxQueue; // Rx buffer size, in bytes<BR> DWORD dwProvSpec1; // provider-specific data<BR> DWORD dwProvSpec2; // provider-specific data<BR> WCHAR wcProvChar[1]; // provider-specific data<BR> } COMMPROP;<BR> dwMaxBaud:<BR> BAUD_075 75 bps<BR> BAUD_110 110 bps<BR> BAUD_134_5 134.5 bps<BR> BAUD_150 150 bps<BR> BAUD_300 300 bps<BR> BAUD_600 600 bps<BR> BAUD_1200 1200 bps<BR> BAUD_1800 1800 bps<BR> BAUD_2400 2400 bps<BR> BAUD_4800 4800 bps<BR> BAUD_7200 7200 bps<BR> BAUD_9600 9600 bps<BR> BAUD_14400 14400 bps<BR> BAUD_19200 19200 bps<BR> BAUD_38400 38400 bps<BR> BAUD_56K 56K bps<BR> BAUD_57600 57600 bps<BR> BAUD_115200 115200 bps<BR> BAUD_128K 128K bps<BR> BAUD_USER Programmable baud rates available<BR> dwProvSubType:<BR> PST_FAX 传真设备<BR> PST_LAT LAT协议<BR> PST_MODEM 调制解调器设备<BR> PST_NETWORK_BRIDGE 未指定的网桥<BR> PST_PARALLELPORT 并口<BR> PST_RS232 RS-232口<BR> PST_RS422 RS-422口<BR> PST_RS423 RS-432口<BR> PST_RS449 RS-449口<BR> PST_SCANNER 扫描仪设备<BR> PST_TCPIP_TELNET TCP/IP Telnet协议<BR> PST_UNSPECIFIED 未指定<BR> PST_X25 X.25标准<BR> dwProvCapabilities<BR> PCF_16BITMODE 支持特殊的16位模式<BR> PCF_DTRDSR 支持DTR(数据终端就绪)/DSR(数据设备就绪)<BR> PCF_INTTIMEOUTS 支持区间超时<BR> PCF_PARITY_CHECK 支持奇偶校验<BR> PCF_RLSD 支持RLSD(接收线信号检测)<BR> PCF_RTSCTS 支持RTS(请求发送)/CTS(清除发送)<BR> PCF_SETXCHAR 支持可设置的XON/XOFF<BR> PCF_SPECIALCHARS 支持特殊字符<BR> PCF_TOTALTIMEOUTS 支持总(占用时间)超时<BR> PCF_XONXOFF 支持XON/XOFF流控制<BR> 标准RS-232和WINDOW支持除PCF_16BITMODE和PCF_SPECIALCHAR外的所有功能<BR> dwSettableParams<BR> SP_BAUD 可配置波特率<BR> SP_DATABITS 可配置数据位个数<BR> SP_HANDSHAKING 可配置握手(流控制)<BR> SP_PARITY 可配置奇偶校验模式<BR> SP_PARITY_CHECK 可配置奇偶校验允许/禁止<BR> SP_RLSD 可配置RLSD(接收信号检测)<BR> SP_STOPBITS 可配置停止位个数<BR> 标准RS-232和WINDOW支持以上所有功能<BR> wSettableData<BR> DATABITS_5 5个数据位<BR> DATABITS_6 6个数据位<BR> DATABITS_7 7个数据位<BR> DATABITS_8 8个数据位<BR> DATABITS_16 16个数据位<BR> DATABITS_16X 通过串行硬件线路的特殊宽度路径<BR> WINDOWS 95支持16的所有设置<BR><BR><B>5.DCB结构:</B><BR><BR> typedef struct _DCB {// dcb<BR> DWORD DCBlength; // sizeof(DCB)<BR> DWORD BaudRate; // current baud rate<BR> 指定当前的波特率<BR> DWORD fBinary: 1; // binary mode, no EOF check<BR> 指定是否允许二进制模式,<BR> WINDOWS 95中必须为TRUE<BR> DWORD fParity: 1; // enable parity checking<BR> 指定奇偶校验是否允许<BR> DWORD fOutxCtsFlow:1; // CTS output flow control<BR> 指定CTS是否用于检测发送控制。<BR> 当为TRUE是CTS为OFF,发送将被挂起。<BR> DWORD fOutxDsrFlow:1; // DSR output flow control<BR> 指定CTS是否用于检测发送控制。<BR> 当为TRUE是CTS为OFF,发送将被挂起。<BR> DWORD fDtrControl:2; // DTR flow control type<BR> DTR_CONTROL_DISABLE值将DTR置为OFF, DTR_CONTROL_ENABLE值将DTR置为ON, DTR_CONTROL_HANDSHAKE允许DTR"握手",DWORD fDsrSensitivity:1; // DSR sensitivity 当该值为TRUE时DSR为OFF时接收的字节被忽略<BR> DWORD fTXContinueOnXoff:1; // XOFF continues Tx<BR> 指定当接收缓冲区已满,并且驱动程序已经发<BR> 送出XoffChar字符时发送是否停止。<BR> TRUE时,在接收缓冲区接收到缓冲区已满的字节XoffLim且驱动程序已经发送出XoffChar字符中止接收字节之后,发送继续进行。<BR> FALSE时,在接收缓冲区接收到代表缓冲区已空的字节XonChar且驱动程序已经发送出恢复发送的XonChar之后,发送继续进行。<BR> DWORD fOutX: 1; // XON/XOFF out flow control<BR> TRUE时,接收到XoffChar之后便停止发送<BR> 接收到XonChar之后将重新开始<BR> DWORD fInX: 1; // XON/XOFF in flow control<BR> TRUE时,接收缓冲区接收到代表缓冲区满的XoffLim之后,XoffChar发送出去<BR> 接收缓冲区接收到代表缓冲区空的XonLim之后,XonChar发送出去<BR> DWORD fErrorChar: 1; // enable error replacement<BR> 该值为TRUE且fParity为TRUE时,用ErrorChar 成员指定的字符代替奇偶校验错误的接收字符<BR> DWORD fNull: 1; // enable null stripping<BR> TRUE时,接收时去掉空(0值)字节<BR> DWORD fRtsControl:2; // RTS flow control<BR> RTS_CONTROL_DISABLE时,RTS置为OFF<BR> RTS_CONTROL_ENABLE时, RTS置为ON<BR> RTS_CONTROL_HANDSHAKE时,<BR> 当接收缓冲区小于半满时RTS为ON<BR> 当接收缓冲区超过四分之三满时RTS为OFF<BR> RTS_CONTROL_TOGGLE时,<BR> 当接收缓冲区仍有剩余字节时RTS为ON ,否则缺省为OFF<BR> DWORD fAbortOnError:1; // abort reads/writes on error<BR> TRUE时,有错误发生时中止读和写操作<BR> DWORD fDummy2:17; // reserved<BR> 未使用<BR> WORD wReserved; // not currently used<BR> 未使用,必须为0<BR> WORD XonLim; // transmit XON threshold<BR> 指定在XON字符发送这前接收缓冲区中可允许的最小字节数<BR> WORD XoffLim; // transmit XOFF threshold<BR> 指定在XOFF字符发送这前接收缓冲区中可允许的最小字节数<BR> BYTE ByteSize; // number of bits/byte, 4-8<BR> 指定端口当前使用的数据位<BR> BYTE Parity; // 0-4=no,odd,even,mark,space<BR> 指定端口当前使用的奇偶校验方法,可能为:<BR> EVENPARITY,MARKPARITY,NOPARITY,ODDPARITY<BR> BYTE StopBits; // 0,1,2 = 1, 1.5, 2<BR> 指定端口当前使用的停止位数,可能为:<BR> ONESTOPBIT,ONE5STOPBITS,TWOSTOPBITS<BR> char XonChar; // Tx and Rx XON character<BR> 指定用于发送和接收字符XON的值<BR> char XoffChar; // Tx and Rx XOFF character<BR> 指定用于发送和接收字符XOFF值<BR> char ErrorChar; // error replacement character<BR> 本字符用来代替接收到的奇偶校验发生错误时的值<BR> char EofChar; // end of input character<BR> 当没有使用二进制模式时,本字符可用来指示数据的结束<BR> char EvtChar; // received event character<BR> 当接收到此字符时,会产生一个事件<BR> WORD wReserved1; // reserved; do not use 未使用<BR> } DCB;<BR><BR><B>6.改变端口设置</B><BR><BR> 使用如下的两个方法<BR> BOOL GetCommState(hComm,&dcb);<BR> BOOL SetCommState(hComm,&dcb);<BR><BR><B>7.改变普通设置</B><BR><BR> BuildCommDCB(szSettings,&DCB);<BR> szSettings的格式:baud parity data stop<BR> 例: "baud=96 parity=n data=8 stop=1"<BR> 简写:"96,N,8,1"<BR> szSettings 的有效值<BR> baud:<BR> 11 or 110 = 110 bps<BR> 15 or 150 = 150 bps<BR> 30 or 300 = 300 bps<BR> 60 or 600 = 600 bps<BR> 12 or 1200 = 1200 bps<BR> 24 or 2400 = 2400 bps<BR> 48 or 4800 = 4800 bps<BR> 96 or 9600 = 9600 bps<BR> 19 or 19200= 19200bps<BR> parity:<BR> n=none<BR> e=even<BR> o=odd<BR> m=mark<BR> s=space<BR> data:<BR> 5,6,7,8<BR> StopBit<BR> 1,1.5,2<BR><BR><B>8.COMMCONFIG结构:</B><BR><BR> typedef struct _COMM_CONFIG {<BR> DWORD dwSize;<BR> WORD wVersion;<BR> WORD wReserved;<BR> DCB dcb;<BR> DWORD dwProviderSubType;<BR> DWORD dwProviderOffset;<BR> DWORD dwProviderSize;<BR> WCHAR wcProviderData[1];<BR> } COMMCONFIG, *LPCOMMCONFIG;<BR> 可方便的使用BOOL CommConfigDialog(<BR> LPTSTR lpszName,<BR> HWND hWnd,<BR> LPCOMMCONFIG lpCC);<BR> 来设置串行口。<BR><BR><B>9.超时设置:</B><BR><BR> 可通过COMMTIMEOUTS结构设置超时,<BR> typedef struct _COMMTIMEOUTS {<BR> DWORD ReadIntervalTimeout;<BR> DWORD ReadTotalTimeoutMultiplier;<BR> DWORD ReadTotalTimeoutConstant;<BR> DWORD WriteTotalTimeoutMultiplier;<BR> DWORD WriteTotalTimeoutConstant;<BR> } COMMTIMEOUTS,*LPCOMMTIMEOUTS;<BR> 区间超时:(仅对从端口中读取数据有用)它指定在读取两个字符之间要经历的时间<BR> 总超时: 当读或写特定的字节数需要的总时间超过某一阈值时,超时触发.<BR> 超时公式:<BR> ReadTotalTimeout = (ReadTotalTimeoutMultiplier * bytes_to_read)<BR> + ReadToTaltimeoutConstant<BR> WriteTotalTimeout = (WriteTotalTimeoutMuliplier * bytes_to_write)<BR> + WritetoTotalTimeoutConstant<BR> NOTE:在设置超时时参数0为无限等待,既无超时<BR> 参数MAXDWORD为立即返回<BR> 超时设置:<BR> GetCommTimeouts(hComm,&timeouts);<BR> SetCommTimeouts(hComm,&timeouts);<BR><BR><B>10.查询方式读写数据</B><BR><BR> <B> 例程:</B><BR> COMMTIMEOUTS to;<BR> DWORD ReadThread(LPDWORD lpdwParam)<BR> {<BR> BYTE inbuff[100];<BR> DWORD nBytesRead;<BR> if(!(cp.dwProvCapabilities&PCF_INTTIMEOUTS))<BR> return 1L;<BR> memset(&to,0,sizeof(to));<BR> to.ReadIntervalTimeout = MAXDWORD;<BR> SetCommTimeouts(hComm,&to);<BR> while(bReading)<BR> {<BR> if(!ReadFile(hComm,inbuff,100,&nBytesRead,NULL))<BR> locProcessCommError(GetLastError());<BR> else<BR> if(nBytesRead)<BR> locProcessBytes(inbuff,nBytesRead);<BR> }<BR> PurgeComm(hComm,PURGE_RXCLEAR);<BR> return 0L;<BR> }<BR> NOTE:<BR> PurgeComm()是一个清除函数,它可以中止任何未决的后台读或写,并且可以冲掉I/O缓冲区.<BR> BOOL PurgeComm(HANDLE hFile,DWORD dwFlags);<BR> dwFlages的有效值:<BR> PURGE_TXABORT: 中止后台写操作<BR> PRUGE_RXABORT: 中止后台读操作<BR> PRUGE_TXCLEAR: 清除发送缓冲区<BR> PRUGE_RXCLEAR: 清除接收缓冲区<BR><B> 技巧:</B><BR> 可通过ClearCommError()来确定接收缓区中处于等待的字节数。<BR> BOOL ClearCommError(<BR> HANDLE hFile, // handle to communications device<BR> LPDWORD lpErrors, // pointer to variable to receive error codes<BR> LPCOMSTAT lpStat // pointer to buffer for communications status<BR> );<BR> ClearCommError()将返回一个COMSTAT结构:<BR> typedef struct _COMSTAT { // cst<BR> DWORD fCtsHold : 1; // Tx waiting for CTS signal<BR> DWORD fDsrHold : 1; // Tx waiting for DSR signal<BR> DWORD fRlsdHold : 1; // Tx waiting for RLSD signal<BR> DWORD fXoffHold : 1; // Tx waiting, XOFF char rec`d<BR> DWORD fXoffSent : 1; // Tx waiting, XOFF char sent<BR> DWORD fEof : 1; // EOF character sent<BR> DWORD fTxim : 1; // character waiting for Tx<BR> DWORD fReserved : 25; // reserved<BR> DWORD cbInQue; // bytes in input buffer<BR> DWORD cbOutQue; // bytes in output buffer<BR> } COMSTAT, *LPCOMSTAT;<BR> 其中的cbInQue和cbOutQue中即为缓冲区字节。<BR><BR><B>11.同步I/O读写数据</B><BR><BR> COMMTIOMOUTS to;<BR> DWORD ReadThread(LPDWORD lpdwParam)<BR> {<BR> BYTE inbuff[100];<BR> DWORD nByteRead,dwErrorMask,nToRead;<BR> COMSTAT comstat;<BR> if(!cp.dwProvCapabilities&PCF_TOTALTIMEOUTS)<BR> return 1L;<BR> memset(&to,0,sizeof(to));<BR> to.ReadTotalTimeoutMultiplier = 5;<BR> to.ReadTotalTimeoutConstant = 50;<BR> SetCommTimeouts(hComm,&to);<BR> while(bReading)<BR> {<BR> ClearCommError(hComm,&dwErrorMask,&comstat);<BR> if(dwErrorMask)<BR> locProcessCommError(dwErrorMask);<BR> if(comstat.cbInQue >100)<BR> nToRead = 100;<BR> else<BR> nToRead = comstat.cbInQue;<BR> if(nToRead == 0)<BR> continue;<BR> if(!ReadFile(hComm,inbuff,nToRead,&nBytesRead,NULL))<BR> locProcessCommError(GetLastError());<BR> else<BR> if(nBytesRead)<BR> locProcessBytes(inbuff,nBytesRead);<BR> }<BR> return 0L;<BR> }<BR><BR><B>12.异步I/O读写数据</B><BR><BR> 当CreateFile()中的fdwAttrsAndFlags参数为FILE_FLAG_OVERLAPPEN时, 端口是为异步I/O打开的,此时可以在ReadFile的最后一个参数中指定一个OVERLAPPED结构,使数据的读操作在后台进行。WINDOWS 95包括了异步I/O的许多变种。<BR> typedef struct _OVERLAPPED {<BR> DWORD Internal;<BR> DWORD InternalHigh;<BR> DWORD Offset;<BR> DWORD OffsetHigh;<BR> HANDLE hEvent;<BR> } OVERLAPPED;<BR> 对于串行口仅hEvent成员有效,其于成员必须为0。<BR> 例程:<BR> COMMTIMEOUTS to;<BR> ...<BR> DWORD ReadThread((LPDWORD lpdwParam)<BR> {<BR> BYTE inbuff[100];<BR> DWORD nRytesRead,endtime,lrc;<BR> static OVERLAPPED o;<BR> if(!cp.dwProvCapabilities & PCF_TOTALTIMEOUTS)<BR> return 1L;<BR> memset(&to,0,sizeof(to));<BR> to.ReadTotalTimeoutMultiplier = 5;<BR> to.ReadTotalTimeoutConstant = 1000;<BR> SetCommTimeouts(hComm,&to);<BR> o.hEvent = CreateEvent(NULL,TRUE,FALSE,NULL);<BR> while(bReading)<BR> {<BR> if(!ReadFile(hComm,inbuff,10,&nBytesRead,&o))<BR> {<BR> nBytesRead = 0;<BR> if(lrc=GetLastError() == ERROR_IO_PENDING)<BR> {<BR> endtime = GetTickCount() + 1000;<BR> while(!GetOverlappedResult(hComm,&o,&nBytesRead,FALSE))<BR> if(GetTickCount() > endtime) break;<BR> }<BR> if(nBytesRead) locProcessBytes(inbuff,nBytesRead);<BR> }<BR> else<BR> {<BR> if(nBytesRead) locProcessBytes(inbuff,nBytesRead);<BR> ResetEvent(o.hEvent);<BR> }<BR> }<BR> PurgeComm(hComm,PURGE_RXCLEAR);<BR> return 0L;<BR> }<BR> 这一例程是对一开始读缓冲区就读到所需的字节时的处理:<BR> while(bReading)<BR> {<BR> if(!ReadFile(hComm,inbuff,10,&nBytesRead,&o))<BR> {<BR> if((lrc=GetLastError()) ==ERROR_IO_PENDING)<BR> {<BR> if(GetOverlappedResult(hComm,&o,&nBytesRead,TRUE))<BR> {<BR> if(nBytesRead)<BR> locProcessBytesa(inbuff,nBytesRead);<BR> }<BR> else<BR> locProcessCommError(GetLastError());<BR> }<BR> else<BR> locProcessCommError(GetLastError));<BR> }<BR> else<BR> if(nBytesRead) locProcessBytes(inbuff,nBytesRead);<BR> ResetEvent(o.hEvent);<BR> }<BR><BR><B>13.事件驱I/O读写:</B><BR><BR> GetCommMask(hComm,&dwMask)<BR> Windows 95报告给应用程序的事件由此方法返回。<BR> SetCommMasl(hComm,&dwMask)<BR> 添加或修改Windows 95所报告的事件列表。<BR> 事件掩码如下:<BR> EV_BREAK 检测到输入为止<BR> EV_CTS CTS(清除发送)信号改变状态<BR> EV_DSR DSR(数据设置就绪)信号改变状态<BR> EV_ERR 发生了线路状态错误.<BR> 线路状态错误为:<BR> CE_FRAME(帧错误)<BR> CE_OVERRUN(接收缓冲区超限)<BR> CE_RXPARITY(奇偶校验错误)<BR> EV_RING 检测到振铃<BR> EV_RLSD RLSD(接收线路信号检测)信号改变状态<BR> EV_EXCHAR 接收到一个字符,并放入输入缓冲区<BR> EV_RXFLAG 接收到事件字符(DCB成员的EvtChar成员),度放入输入缓冲区<BR> EV_TXEMPTY 输出缓冲区中最后一个字符发送出去<BR> 在用SetCommMask指定了有用的事件后,应用程序可调用WaitCommEvent()来等待事件发生.<BR> BOOL WaitCommEvent(<BR> HANDLE hFile, // handle of communications device<BR> LPDWORD lpEvtMask, // address of variable for event that occurred<BR> LPOVERLAPPED lpOverlapped, // address of overlapped structure<BR> );<BR> 此方法可以以同步或异步方式操作<BR> 例程:<BR> COMMTIMEOUTS to;<BR> ...<BR> DWORD ReadTherad(LPDWORD lpdwParam)<BR> {<BR> BYTE binbuff[100];<BR> DWORD nBytesRead,dwEvent,dwError;<BR> COMSTAT cs;<BR> SetCommMask(hComm,EV_RXHAR);<BR> while(bReading)<BR> {<BR> if(WaitCommEvent(hComm,&dwEvent,NULL))<BR> {<BR> ClearCommError(hComm,&dwError,&cs);<BR> if((dwEvent&EV_RXCHAR)&&cs.cbInQue)<BR> {<BR> if(!ReadFile(hComm,inbuff,cs.cbInQue,&nBytesRead,NULL)<BR> locProcessCommError(GetLastError());<BR> }<BR> else<BR> {<BR> if(nByteRead)<BR> locProcessBytes(inbuff,nBytesRead);<BR> }<BR> else<BR> locProcessCommError(GetLastError());<BR> }<BR> PurgeComm(hComm,PURGE_RXCLEAR);<BR> return 0L;<BR> }<BR> NOTE: SetCommMask(hComm,0)可使WaitCommEvent()中止.<BR> 可使用GetCommmodemStatus()方法,例程:<BR> if(cp.dwProvCapabilities&PCF_RTSCTS)<BR> {<BR> SetCommMask(hComm,EV_CTS);<BR> WaitCommEvent(hComm,&dwMask,NULL);<BR> if(dwMask&EV_CTS)<BR> {<BR> GetCommModemStatus(hComm,&dwStatus)<BR> if(dwStatus&MS_CTS_ON) /* CTS stransition OFF-ON */<BR> else /* CTS stransition ON-OFF */<BR> }<BR> }<BR> MS_CTS_ON CTS为ON<BR> MS_DSR_ON DSR为ON<BR> MS_RING_ON RING为ON<BR> MS_ELSD_ON RLSD为ON<BR><BR><B>14.错误</B><BR><BR> 当发生错误时应用方法ClearCommError(hComm,&dwErrorMask,&constat)得到错误掩码。<BR> CE_BREAK 中止条件<BR> CE_FRAME 帧错误<BR> CW_IOE 一般I/O错误,常伴有更为详细的错误标志<BR> CE_MODE 不支持请求的模式<BR> CE_OVERRUN 缓冲区超限下一个字符将丢失<BR> CE_RXOVER 接收缓冲区超限<BR> CE_RXPARITY 奇偶校验错误<BR> CE_TXFULL 发送缓冲区满<BR> CE_DNS 没有选择并行设备<BR> CE_PTO 并行设备发生超时<BR> CE_OOP 并行设备缺纸<BR><BR><B>15.控制命令</B><BR><BR> EscapeCommFunction()可将硬件信号置ON或OFF,模拟XON或XOFF</FONT></P>
<P><SPAN class=line><FONT color=#386490> BOOL EscapeCommFunction(<BR> HANDLE hFile, // handle to communications device<BR> DWORD dwFunc // extended function to perform<BR> );<BR> dwFunc的有效值(可用'|'同时使用多个值)<BR> CLRDTR DTR置OFF<BR> CLRRTS RTS置OFF<BR> SETDTR STR置ON<BR> SETRTS TRS置ON<BR> SETXOFF 模拟XOFF字符的接收<BR> SETXON 模拟XON字符的接收<BR> SETBREAK 在发送中产生一个中止<BR> CLRBREAK 在发送中清除中止<BR></FONT></SPAN></P> </td>
</tr>
<tr>
<td colspan="2" align='right' class="left_tdbgall">文章录入:<a href='/UserInfo.asp?UserName=admin'>admin</a> 责任编辑:admin </td>
</tr>
<tr>
<td width="5"></td>
<td width="752"><li>上一篇文章: <a href='/Article/Interface/COM/COMPRG/200411/56.html' title='文章标题:VC中ODBC数据库技术应用源程序详解之二
作 者:龚建伟等
更新时间:2004-11-24 下午 05:32:52'>VC中ODBC数据库技术应用源程序详解之二</a></li><br>
<li>下一篇文章: <a href='/Article/Interface/COM/COMPRG/200411/58.html' title='文章标题:VC中基于文档程序中使用MSComm通讯控件问题
作 者:龚建伟等
更新时间:2004-11-24 下午 05:37:32'>VC中基于文档程序中使用MSComm通讯控件问题</a></li></td>
</tr>
<tr align="right" class="left_tdbgall">
<td colspan="2">【<a href='/Article/Comment.asp?ArticleID=57' target='_blank'>发表评论</a>】【<a href='/Article/SendMail.asp?ArticleID=57' target='_blank'>告诉好友</a>】【<a href='/Article/Print.asp?ArticleID=57' target='_blank'>打印此文</a>】【<a href='javascript:window.close();'>关闭窗口</a>】 </td>
</tr>
</table>
<!--文章显示代码结束-->
<!--最新热点、最新推荐、相关文章代码开始-->
<table width="760" border="0" cellpadding="0" cellspacing="0" class="center_tdbgall" style="word-break:break-all;Width:fixed">
<tr>
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -