C语言程序周期接收虚拟串口发送的数据_c语言编程实现串口收发数据的程序例程(3)

Windows下这些函数的使用方法可以参考下面两篇博客:


示例

下面我们主要对串口周期发送,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的周期发送

https://raw.githubusercontent.com/xkyvvv/blogpic/main/pic1/image-20210804162510858.png

再次运行程序

请添加图片描述

可以发现,我们的程序成功的接收了串口周期发送的数据。

这边需要注意的是,我们接收的字符串数组需要比我们实际接收的字节数打一个,因为在C语言中,字符串都是以数组的形式存储的,而为了知道字符串什么时候结束,编译器会在字符串尾部增加一个’\0’,因此实际我们接收8个字节,但是接收字符串数组是char str[9]。


探究

上面只是一个很简单很理想的例子,现在我们不妨把问题想复杂一点。

有几个可以值得讨论的设置问题:

  • 周期问题:上面我们程序接收的周期是1000ms,串口发送的周期也是1000ms,为了尽快地获得到所有的数据,它们之间的周期需要满足什么要求;
  • 发送数据大小和接收数据缓存大小:上面我们设置的发送数据大小和接收数据大小是一致的,实际上我们一般不会这样弄,我们一般会把接收数据缓存尽量弄大一点,而且一般我们会使用环形缓冲数据结构来保存数据,我的一篇博客分享了一个C语言环形缓冲库的使用【C语言开源库】在CLion上使用一个轻量的适合嵌入式系统的环形缓冲库ring buffer 和C语言Unity单元测试框架
  • 数据解析问题:因为条件限制的原因,我们虚拟串口周期发送的数据是一样的,实际情况可能是发送的不一样的数据,因此我们还需要对接收的数据进行识别解析。

周期问题

我们程序从串口接收数据,肯定是希望能够尽快的接收的所有数据,不能遗漏任何数据,也尽可能的保持实时性。

基于上面的前提,那我们可以得出一些简单的结论:接收的周期必须要小于等于发送的周期,在不考虑接收和发送缓存的情况下,我们要想接收到所有的数据,我们就必须接收比发送更频繁。

下面我们测试一下接收周期5000ms,发送周期为1000ms。

在这里插入图片描述

可以看到,接收不到数据。

测试一下接收周期1500ms,发送周期为1000ms。

https://raw.githubusercontent.com/xkyvvv/blogpic/main/pic1/image-20210804171709662.png

也接收不到数据。

测试一下接收周期1200ms,发送周期为1000ms。

https://raw.githubusercontent.com/xkyvvv/blogpic/main/pic1/image-20210804171819742.png

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Go语言工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Go语言全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Golang知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip1024b (备注Go)
img

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

义、实战项目、讲解视频,并且后续会持续更新**

如果你觉得这些内容对你有帮助,可以添加V获取:vip1024b (备注Go)
[外链图片转存中…(img-CcUfviKZ-1712865063212)]

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

  • 7
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值