百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

循环队列原理及在单片机串口通讯中的应用(二)

wptr33 2025-07-06 17:24 31 浏览

前言

书接上回,前文主要介绍了环形队列的实现原理以及C语言实现及测试过程,本文将回归到嵌入式平台的应用中,话不多说,淦,上干货!

实验目的

  • HAL库下串口的配置及使用
  • 环形队列在串口数据接收中的使用

硬件环境

  • falling-star board(自设计,下期开源资料,主控STM32f103RET6)

软件环境

  • keil5
  • cubemx

cubemx配置

1、 时钟的配置,

无论什么平台,什么单片机,第一步,我想都是要搞清楚时钟,时钟是一切的根源,外部晶振选择根据自己的硬件焊接的晶振频率选择,最大频率,此处小飞哥选择的是72MHZ,这个一般越高越好,越高也就意味着功耗越大,当涉及到低功耗的时候,就要考虑不同阶段的时钟频率了。

2、串口配置

主要配置参数见下图:

3、配置调试模式

有时候,我们发现调试模式无法使用,那可能跟这个有关系,通过此配置,我们可以选择不同的模式,同时硬件设计主要注意IO引脚的占用情况。

4、代码生成配置

逻辑代码编写

本次用到的硬件资源不多,cubemx配置也比较少,接下来主要编写环形队列在串口数据处理中的使用。

1、MCU串口接收代码编写

在此之前,先来介绍个串口打印的方法,日常调试过程中,串口打印绝对是必不可少的利器,尤其是在一些安全芯片上,由于没法进行实时仿真,串口打印成了非常简便且有效定位bug的手段,直接看代码:

#include "stdio.h"

#if 1
#pragma import(__use_no_semihosting)
//标准库需要的支持函数
struct __FILE
{
  int handle;
};

FILE __stdout;
//定义_sys_exit()以避免使用半主机模式
void _sys_exit(int x)
{
  x = x;
}
//重定义fputc函数
int fputc(int ch, FILE *f)
{
  while ((USART1->SR & 0X40) == 0)
    ; //循环发送,直到发送完
  USART1->DR = (uint8_t)ch;
  return ch;
}
#endif

#define useruart_printf_debug//产品调试完成后,只需要屏蔽此条宏定义,即可关闭串口打印功能

#ifdef useruart_printf_debug

#define uart_printf(format, args...) \
    do                               \
    {                                \
        printf(format, ##args);      \
    } while (0)
#else
#define uart_printf(format, args...) \
    do                               \
    {                                \
    } while (0)
#endif

串口接收中断编写:

void UserUartInit(void)
{
    HAL_UART_Receive_IT(&huart1, &UART_TXRX_Para.RxData, 1);
    /*初始化顺序循环队列*/
    InitQueue(&Q); 
}
//串口接收回调函数
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
    if (huart->Instance == USART1)
    {
        //数据插入队列中
        EnQueue(&Q, (DataType)UART_TXRX_Para.RxData);
    }
    //接收完一字节需要再次打开接收中断
    HAL_UART_Receive_IT(&huart1, &UART_TXRX_Para.RxData, 1);
}

串口接收数据处理,这部分比较简单,我们就在while中调用,有数据就去取出去,然后串口发送出来

void SCQueue_MessageRecive(void)
{
    uint8_t data_temp;
    int data_len = 0;

//    data_len = QueueLength(&Q);
//    if (data_len)
  if(!QueueEmpty(Q))
    { 
    DeQueue(&Q, &data_temp);
        HAL_UART_Transmit(&huart1, &data_temp, 1, 100);
    }
}

2、环形队列数据处理测试

  • 附加标志法:

为了更好的演示“转圈圈的效果”,我们先来写入,不进行读取,看看会发生什么事情:开始初始化队列为空,然后我们写入数据,当我们写入52字节数据时,实际入队列30字节,队列满后便不再接收数据。

然后我们通过按键控制,每次读取一个字节,读取出几个字节,使得队列出于未满状态,可以看到队列头指针不断增加,不断追赶尾指针,满标志为0,当头指针小于尾指针的时候,是可以继续插入数据的。

那当头指针追上尾指针的时候会发生什么事情呢?当数据全部取出的时候,头、尾指针相同,队列重新处于空状态,完成一个圆圈,当然这里是为了效果更明显,写满,全部读出,实际不会这样,队列的空间会比实际接受的数据大一些,不断存储、读取,想成“你追我赶”的追逐游戏,直到一帧数据结束。

在实际使用过程中,为了加快数据处理速度,我们希望是能边写入边读取的,这样效率要比完全接收完成之后再做处理节省不少时间,接下来,进行测试边存储边读取的效果,理想的是应该在一个环里不断转圈(添加视频):

  • 预留空间法

预留空间法与附加标志法大致相同,区别在于,会有一个单位空间剩余作为判断满队列用,其他的过程都是一样的,就不做介绍了,主要看下区别,可以看到,最后一字节地址并未写入数据,但提示空间已满,无法继续写入,这也就是预留的一单位空间用于确定队列满状态,但也是会造成空间的浪费。

视频演示:

关于循环队列在单片机中的使用就要到这里了,感谢各位小伙伴耐心阅读,记得点赞,转发哦!

相关推荐

MySQL进阶五之自动读写分离mysql-proxy

自动读写分离目前,大量现网用户的业务场景中存在读多写少、业务负载无法预测等情况,在有大量读请求的应用场景下,单个实例可能无法承受读取压力,甚至会对业务产生影响。为了实现读取能力的弹性扩展,分担数据库压...

Postgres vs MySQL_vs2022连接mysql数据库

...

3分钟短文 | Laravel SQL筛选两个日期之间的记录,怎么写?

引言今天说一个细分的需求,在模型中,或者使用laravel提供的EloquentORM功能,构造查询语句时,返回位于两个指定的日期之间的条目。应该怎么写?本文通过几个例子,为大家梳理一下。学习时...

一文由浅入深带你完全掌握MySQL的锁机制原理与应用

本文将跟大家聊聊InnoDB的锁。本文比较长,包括一条SQL是如何加锁的,一些加锁规则、如何分析和解决死锁问题等内容,建议耐心读完,肯定对大家有帮助的。为什么需要加锁呢?...

验证Mysql中联合索引的最左匹配原则

后端面试中一定是必问mysql的,在以往的面试中好几个面试官都反馈我Mysql基础不行,今天来着重复习一下自己的弱点知识。在Mysql调优中索引优化又是非常重要的方法,不管公司的大小只要后端项目中用到...

MySQL索引解析(联合索引/最左前缀/覆盖索引/索引下推)

目录1.索引基础...

你会看 MySQL 的执行计划(EXPLAIN)吗?

SQL执行太慢怎么办?我们通常会使用EXPLAIN命令来查看SQL的执行计划,然后根据执行计划找出问题所在并进行优化。用法简介...

MySQL 从入门到精通(四)之索引结构

索引概述索引(index),是帮助MySQL高效获取数据的数据结构(有序),在数据之外,数据库系统还维护者满足特定查询算法的数据结构,这些数据结构以某种方式引用(指向)数据,这样就可以在这些数据结构...

mysql总结——面试中最常问到的知识点

mysql作为开源数据库中的榜一大哥,一直是面试官们考察的重中之重。今天,我们来总结一下mysql的知识点,供大家复习参照,看完这些知识点,再加上一些边角细节,基本上能够应付大多mysql相关面试了(...

mysql总结——面试中最常问到的知识点(2)

首先我们回顾一下上篇内容,主要复习了索引,事务,锁,以及SQL优化的工具。本篇文章接着写后面的内容。性能优化索引优化,SQL中索引的相关优化主要有以下几个方面:最好是全匹配。如果是联合索引的话,遵循最...

MySQL基础全知全解!超详细无废话!轻松上手~

本期内容提醒:全篇2300+字,篇幅较长,可搭配饭菜一同“食”用,全篇无废话(除了这句),干货满满,可收藏供后期反复观看。注:MySQL中语法不区分大小写,本篇中...

深入剖析 MySQL 中的锁机制原理_mysql 锁详解

在互联网软件开发领域,MySQL作为一款广泛应用的关系型数据库管理系统,其锁机制在保障数据一致性和实现并发控制方面扮演着举足轻重的角色。对于互联网软件开发人员而言,深入理解MySQL的锁机制原理...

Java 与 MySQL 性能优化:MySQL分区表设计与性能优化全解析

引言在数据库管理领域,随着数据量的不断增长,如何高效地管理和操作数据成为了一个关键问题。MySQL分区表作为一种有效的数据管理技术,能够将大型表划分为多个更小、更易管理的分区,从而提升数据库的性能和可...

MySQL基础篇:DQL数据查询操作_mysql 查

一、基础查询DQL基础查询语法SELECT字段列表FROM表名列表WHERE条件列表GROUPBY分组字段列表HAVING分组后条件列表ORDERBY排序字段列表LIMIT...

MySql:索引的基本使用_mysql索引的使用和原理

一、索引基础概念1.什么是索引?索引是数据库表的特殊数据结构(通常是B+树),用于...