Windows下这些函数的使用方法可以参考下面两篇博客:
- [CreateFile函数详解](https://www.cnblogs.com/findumars/p/5636108.html)
- 串口之ReadFile、WriteFile函数详解
示例
下面我们主要对串口周期发送,C语言程序这边就周期接收发送的数据。
VSPD创建两个虚拟串口
串口调试助手打开串口2(COM2),串口参数为:波特率115200,数据位8,检验位0,停止位1。
C语言代码为
#include <Windows.h>
#include <stdio.h>
HANDLE hCom;
int main(void)
{
hCom = CreateFile(TEXT(“com1”),//COM1口
GENERIC_READ | GENERIC_WRITE, //允许读和写
0, //指定共享属性,由于串口不能共享,所以该参数必须为0
NULL,
OPEN_EXISTING, //打开而不是创建
FILE_ATTRIBUTE_NORMAL, //属性描述,该值为FILE_FLAG_OVERLAPPED,表示使用异步I/O,该参数为0,表示同步I/O操作
NULL);
if (hCom == INVALID_HANDLE_VALUE)
{
printf(“打开COM失败!\n”);
return FALSE;
}
else
{
printf(“COM打开成功!\n”);
}
SetupComm(hCom, 1024, 1024); //输入缓冲区和输出缓冲区的大小都是1024
/超时设置*****/
COMMTIMEOUTS TimeOuts;
//设定读超时
TimeOuts.ReadIntervalTimeout = MAXDWORD;//读间隔超时
TimeOuts.ReadTotalTimeoutMultiplier = 0;//读时间系数
TimeOuts.ReadTotalTimeoutConstant = 0;//读时间常量
//设定写超时
TimeOuts.WriteTotalTimeoutMultiplier = 1;//写时间系数
TimeOuts.WriteTotalTimeoutConstant = 1;//写时间常量
SetCommTimeouts(hCom, &TimeOuts); //设置超时
/**************配置串口/
DCB dcb;
GetCommState(hCom, &dcb);
dcb.BaudRate = 115200; //波特率为115200
dcb.ByteSize = 8; //每个字节有8位
dcb.Parity = NOPARITY; //无奇偶校验位
dcb.StopBits = ONESTOPBIT; //一个停止位
SetCommState(hCom, &dcb);
DWORD wCount;//实际读取的字节数
bool bReadStat;
char str[8] = { 0 };
while (1)
{
//PurgeComm(hCom, PURGE_TXCLEAR | PURGE_RXCLEAR); //清空缓冲区
bReadStat = ReadFile(hCom, str, sizeof(str), &wCount, NULL);
if (!bReadStat)
{
printf(“读串口失败!”);
return FALSE;
}
else
{
if(wCount != 8) {
printf(“没读取到8个字节数据\n!”);
}
else {
//str[1] = ‘\0’;
printf(“%s\n”, str);
}
}
Sleep(1000); //延时1000ms
}
CloseHandle(hCom);//关闭串口
}
我们定义了一个char str[9]来接收数据,首先通过ReadFile()函数的返回值判断读串口是否成功,如果成功,则通过wCount来判断实际接收到的数据是否和我们预想的一致。
我们在串口2那先不发送数据,直接运行,结果如下
可以发现是接收不到数据的,因此会一直打印没有接收到8个字节数据。
我们设置一下串口2的周期发送
再次运行程序
可以发现,我们的程序成功的接收了串口周期发送的数据。
这边需要注意的是,我们接收的字符串数组需要比我们实际接收的字节数打一个,因为在C语言中,字符串都是以数组的形式存储的,而为了知道字符串什么时候结束,编译器会在字符串尾部增加一个’\0’,因此实际我们接收8个字节,但是接收字符串数组是char str[9]。
探究
上面只是一个很简单很理想的例子,现在我们不妨把问题想复杂一点。
有几个可以值得讨论的设置问题:
- 周期问题:上面我们程序接收的周期是1000ms,串口发送的周期也是1000ms,为了尽快地获得到所有的数据,它们之间的周期需要满足什么要求;
- 发送数据大小和接收数据缓存大小:上面我们设置的发送数据大小和接收数据大小是一致的,实际上我们一般不会这样弄,我们一般会把接收数据缓存尽量弄大一点,而且一般我们会使用环形缓冲数据结构来保存数据,我的一篇博客分享了一个C语言环形缓冲库的使用【C语言开源库】在CLion上使用一个轻量的适合嵌入式系统的环形缓冲库ring buffer 和C语言Unity单元测试框架;
- 数据解析问题:因为条件限制的原因,我们虚拟串口周期发送的数据是一样的,实际情况可能是发送的不一样的数据,因此我们还需要对接收的数据进行识别解析。
周期问题
我们程序从串口接收数据,肯定是希望能够尽快的接收的所有数据,不能遗漏任何数据,也尽可能的保持实时性。
基于上面的前提,那我们可以得出一些简单的结论:接收的周期必须要小于等于发送的周期,在不考虑接收和发送缓存的情况下,我们要想接收到所有的数据,我们就必须接收比发送更频繁。
下面我们测试一下接收周期5000ms,发送周期为1000ms。
可以看到,接收不到数据。
测试一下接收周期1500ms,发送周期为1000ms。
也接收不到数据。
测试一下接收周期1200ms,发送周期为1000ms。
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Go语言工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Go语言全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Golang知识点,真正体系化!
由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新
如果你觉得这些内容对你有帮助,可以添加V获取:vip1024b (备注Go)
一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
义、实战项目、讲解视频,并且后续会持续更新**
如果你觉得这些内容对你有帮助,可以添加V获取:vip1024b (备注Go)
[外链图片转存中…(img-CcUfviKZ-1712865063212)]
一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!