当前位置:C++技术网 > 资讯 > 使用WinAPI串口编程

使用WinAPI串口编程

更新时间:2015-11-05 22:02:25浏览次数:1+次

串口和其他通信设备是作为文件处理的。我们可以用MSComm控件来实现,这个比较简单。我们也可以用事件驱动的方式。所谓事件驱动,就是当有串行数据进入输入数据,自动执行程序。我们使用WinAPI来编写串口通信程序。

首先使用CreateFile函数打开串口,以非重叠的方式打开。该函数返回一个串口句柄,使用该句柄初始化串口参数,然后使用WriteFile函数发送串行数据,使用ReadFile函数从串口读取数据。最后就是关闭我们的串口句柄利用CloseHandle()。


HANDLE WINAPI CreateFile(
_In_ LPCTSTR lpFileName,
_In_ DWORD dwDesiredAccess,
_In_ DWORD dwShareMode,
_In_opt_ LPSECURITY_ATTRIBUTES lpSecurityAttributes,
_In_ DWORD dwCreationDisposition,
_In_ DWORD dwFlagsAndAttributes,
_In_opt_ HANDLE hTemplateFile
);
使用该函数可以打开一个串口,并返回一个句柄,在程序中便可以通过访问该句柄来实现对串口的访问。
lpFileName:用字符串来标识打开的串口编号、如"COM1"表示串口1。
dwDesiredAccess:对串口的操作方式。对串口的操作通常既有读输入缓冲区操作,也有写缓冲区操作。所以通常该函数设为:
GENERIC_READ | GENERIC_WRITE;
dwShareMode:共享模式设置。串口不允许共享的设备,所以该参数为o
lpSecurityAttributes:选择默认值就行
dwCreationDisposition:对于串口通信,只能选择OPEN_EXISTING,即打开现有的串口。
dwFlagsAndAttributes:对于串口,该参数只能设置为"FILE_ATTRIBUTE_NORMAL"(非重叠操作)
hTemplateFile:该函数设为NULL就行。
SetupComm()函数
设置串口输入缓冲区及输出缓存区的大小。
BOOL SetupComm(
HANDLE hFile,
DWORD dwInQueue,
DWORD dwOutQueue
);
hFile:串口句柄
dwInQueue:输入缓冲区大小,以字节为单位
dwOutQueue:输出缓冲区大小,以字节为单位
typedef struct _DCB {// dcb
DWORD DCBlength; // sizeof(DCB)
DORD BaudRate; // current baud rate 指定当前的波特率
DWORD fBinary: 1; // binary mode, no EOF check 指定是否允许二进制模式WIN95中须为TRUE      
DWORD fParity: 1; // enable parity checking 指定奇偶校验是否允许
DWORD fOutxCtsFlow:1; // CTS output flow control 指定CTS是否用于检测发送控制。                                                               当为TRUE是CTS为OFF,发送将被挂起。
DWORD fOutxDsrFlow:1; // DSR output flow control         指定DSR是否用于检测发送控制。当为TRUE是DSR为OFF,发送将被挂起。
DWORD fDtrControl:2; // DTR flow control type   DTR_CONTROL_DISABLE值将DTR置为OFF, DTR_CONTROL_ENABLE值将DTR置为ON, DTR_CONTROL_HANDSHAKE 允许DTR"握手",
DWORD fDsrSensitivity:1;// DSR sensitivity 当该值为TRUE时DSR为OFF时接收的字节被忽略
DWORD fTXContinueOnXoff:1; // XOFF continues Tx
              指定当接收缓冲区已满,并且驱动程序已经发送出XoffChar字符时发送是否停止。
            TRUE时,在接收缓冲区接收到缓冲区已满的字节XoffLim且驱动程序已经发送出XoffChar字                    符中止接收字节之后,发送继续进行。
            FALSE时,在接收缓冲区接收到代表缓冲区已空的字节XonChar且驱动程序已经发送出恢复                    发送的XonChar之后,发送继续进行。

DWORD fOutX: 1; // XON/XOFF out flow control    TRUE时,接收到XoffChar之后便停止发                                                  接收到XonChar之后将重新开始
DWORD fInX: 1; // XON/XOFF in flow controlTRUE时,接收缓冲区接收到代表缓冲区满的XoffLim之后,XoffChar发送出去接收缓冲区接收到代表缓冲区空的XonLim之后,XonChar发送出去
DWORD fErrorChar: 1; // enable error replacement 该值为TRUE且fParity为TRUE时,用ErrorChar 成员指定的字符代替奇偶校验错误的接收字符
DWORD fNull: 1; // enable null stripping  TRUE时,接收时去掉空(0值)字节
DWORD fRtsControl:2; // RTS flow control 
DWORD fAbortOnError:1; // abort reads/writes on error  TRUE时,有错误发生时中止读和写操作
                                     RTS_CONTROL_DISABLE时,RTS置为OFF
                                    RTS_CONTROL_ENABLE时, RTS置为ON
                              RTS_CONTROL_HANDSHAKE时,
                                当接收缓冲区小于半满时RTS为ON
                                当接收缓冲区超过四分之三满时RTS为OFF
                                RTS_CONTROL_TOGGLE时,
                                当接收缓冲区仍有剩余字节时RTS为ON ,否则缺省为OFF

DWORD fDummy2:17; // reserved  未使用
WORD wReserved; // not currently used  未使用,必须为0
WORD XonLim; // transmit XON threshold指定在XON字符发送这前接收缓冲区中可允许的最小字节数
WORD XoffLim; // transmit XOFF threshold指定在XOFF字符发送这前接收缓冲区中可允许的最小字节数
BYTE ByteSize; // number of bits/byte, 4-8  指定端口当前使用的数据位
BYTE Parity; // 0-4=no,odd,even,mark,space 指定端口当前使用的奇偶校验方法,可能为:EVENPARITY,MARKPARITY,NOPARITY,ODDPARITY 
BYTE StopBits;//0,1,2 = 1, 1.5, 2   指定端口当前使用的停止位数,可能为:ONESTOPBIT,ONE5STOPBITS,TWOSTOPBITS
char XonChar; // Tx and Rx XON character  指定用于发送和接收字符XON的值
char XoffChar; // Tx and Rx XOFF character  指定用于发送和接收字符XOFF值
char ErrorChar; // error replacement character本字符用来代替接收到的奇偶校验发生错误时的值
char EofChar; // end of input character 当没有使用二进制模式时,本字符可用来指示数据的结束
char EvtChar; // received event character 当接收到此字符时,会产生一个事件
WORD wReserved1; // reserved; do not use 未使用
} DCB;
GetCommState 读取串口设置(波特率,校验,停止位,数据位等).
函数声明:
BOOL GetCommState(
HANDLE hFile,
LPDCB lpDCB
);
GetCommState函数的第一个参数hFile是由CreateFile函数返回指向已打开串行口的句柄。第二个参数指向设备控制块DCB。如果函数调用成功,则返回值为非0;若函数调用失败,则返回值为0。
当应用程序仅仅需要修改一部分串行口的配置值时,可以通过GetCommState函数获得当前的DCB结构,然后更改参数,再调用SetCommState函数设置修改过的DCB来配置串行口。
函数作用:设置串口设置(波特率,校验,停止位,数据位等).
函数设置COM口的设备控制块:
BOOL SetCommState(
HANDLE hFile,
LPDCB lpDCB
);
SetCommState函数的第一个参数hFile是由CreateFile函数返回指向已打开串行口的句柄。第二个参数指向设备控制块DCB。如果函数调用成功,则返回值为非0;若函数调用失败,则返回值为0。


BOOL WriteFile(
HANDLE  hFile,//文件句柄
LPCVOID lpBuffer,//数据缓存区指针
DWORD   nNumberOfBytesToWrite,//你要写的字节数
LPDWORD lpNumberOfBytesWritten,//用于保存实际写入字节数的存储区域的指针
LPOVERLAPPED lpOverlapped//OVERLAPPED结构体指针
);
HANDLE hFile, 需要写入数据的文件指针,这个指针指向的文件必须是GENERIC_WRITE access 访问属性的文件

LPOVERLAPPED lpOverlapped OVERLAPPED结构体指针,如果文件是以FILE_FLAG_OVERLAPPED方式打开的话,那么这个

指针就不能为NULL

BOOL ReadFile(
    HANDLE hFile,                                    //文件的句柄
    LPVOID lpBuffer,                                //用于保存读入数据的一个缓冲区
    DWORD nNumberOfBytesToRead,    //要读入的字节数
    LPDWORD lpNumberOfBytesRead,    //指向实际读取字节数的指针
    LPOVERLAPPED lpOverlapped
    ////如文件打开时指定了FILE_FLAG_OVERLAPPED,那么必须,用这个参数引用一个特殊的结构。
    //该结构定义了一次异步读取操作。否则,应将这个参数设为NULL
);