一个免锁环形缓冲区的实现

2024-09-02 23:32
文章标签 实现 环形 缓冲区 免锁

本文主要是介绍一个免锁环形缓冲区的实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

面是串口DMA+环形缓冲区的实现,数据收发是异步的,不需要死等。
关于环形缓冲区参考:
1
2
http://blog.csdn.net/jieffantfyan/article/details/53572103

实现原理
程序是在串口中断收发方式的基础上设计的,应用层通过环形缓冲区进行串口数据读取,环形缓冲区作为一级缓存,增加DMA作为二级缓存。相对中断方式这种设计可以减少串口进入中断的次数,尤其是在高速波特率的情况下。由于使用DMA收发数据时,必须预设好发送/接收地址、长度等信息,软件内为DMA开辟了16个字节数组作为缓冲区。当发送环形缓存区内有数据需要发送时,程序将前16字节(如果不足则复制实际长度并重新设置DMA发送长度)复制到DMA发送缓冲区内并启动发送,数据发送完成之后将余下的数据按同样的方法复制到DMA发送缓冲区直至数据发送完成。而对于接收来说,处理比发送麻烦一些,需要两个中断服务程序配合,当DMA接收完成16字节之后,软件将这些数据写入到接收环形缓冲区内,以便应用程序读取。因为软件中将DMA接收缓存长度设置为16字节,即DMA必须连续接收到16字节才会进入中断服务程序,但多数情况下接收到的数据长度不可能全是16字节的倍数,比如MCU只收到10字节之后的一段时间内再也收不到数据了。对于这种情况可以使用定时器配合检测,一旦发现长时间收不到串口的数据则将DMA接收缓冲区的数据全部提取出来。由于STM32单片机提供了空闲中断,我们可以利用这个机制解决上述问题,当产生空闲中断时,意味着已经没有数据接收了,这时候则将DMA接收缓冲内的数据提取出来。

外部接口声明
下面将串口的初始化、读、写接口抽象出来。

/******************************************************************************
 * Copyright (C) 2016, roger
 * All rights reserved.
 *
 * 文件名称: tty.h
 * 摘    要:控制台驱动

 *             
 * 当前版本: 3.0
 * 作    者: roger
 * 完成日期: 2016-09-24
 *             
 * 取代版本: 2.0
 * 原作者  : roger
 * 完成日期: 2015-07-08
 ******************************************************************************/


#ifndef _TTY_H_
#define _TTY_H_

#define TTY_BAUDRATE          115200                    /*波特率 ------------*/
#define TTY_TXBUF_SIZE        256                       /*发送缓冲区长度 -----*/
#define TTY_RXBUF_SIZE        256                       /*接收缓冲区长度 -----*/
#define TTY_DMA_TX_LEN        10                        /*DMA 发送缓冲区 ----*/
#define TTY_DMA_RX_LEN        10                        /*DMA 接收缓冲区 ----*/

#define TTY_USE_DMA           1                         /*启用DMA -----------*/


/* Exported Structs ---------------------------------------------------------*/

typedef struct          
{
    void (*init)(void);                                     /*初始化 --------*/    
    unsigned int (*write)(void *buf, unsigned int len);     /*数据写 --------*/
    unsigned int (*read) (void *buf, unsigned int len);     /*读数据 --------*/
    void (*puts)(const char *str);                          /*输入一个字符串 */
    void (*clr)(void);                                      /*清除接收缓冲区 */
    unsigned int (*buflen)(void);                           /*接收缓冲区的长度*/
    void (*printf)(const char *format, ...);                /*格式化打印 ----*/
}tty_t;


/* Exported variables ------------------------------------------------------- */
extern const tty_t tty;


#endif  
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
接口实现 ##‘
/******************************************************************************
 * Copyright (C) 2016, roger
 * All rights reserved.
 *
 * 文件名称: tty.c
 * 摘    要:打印串口驱动

 *             
 * 当前版本: 3.0
 * 作    者: roger
 * 完成日期: 2016-09-24
 *             
 * 取代版本: 2.0
 * 原作者  : roger
 * 完成日期: 2015-07-08
 ******************************************************************************/

/* Includes ------------------------------------------------------------------*/
#include "tty.h"
#include "ringbuffer.h"
#include "stm32f4xx.h"
#include <stdarg.h>
#include <stdio.h>
#include <string.h>

static unsigned char rxbuf[TTY_TXBUF_SIZE];         /*接收缓冲区 ------------*/
static unsigned char txbuf[TTY_RXBUF_SIZE];         /*发送缓冲区 ------------*/
static ring_buf_t ringbuf_send, ringbuf_recv;       /*收发缓冲区管理 ---------*/

#if TTY_USE_DMA == 1
    static unsigned char dma_tx_buf[TTY_DMA_TX_LEN];/*DMA发送缓冲区 ---------*/
    static unsigned char dma_rx_buf[TTY_DMA_RX_LEN];/*DMA接收缓冲区 ---------*/      
#endif

/*******************************************************************************
 * 函数名称:port_conf
 * 功能描述:打印串口配置(PD8->USART3_TX, PD9->USART3_RX)
 * 输入参数:none
 * 返 回 值:none
 * 作    者:roger.luo
 ******************************************************************************/
static void port_conf(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    /*console串口引脚配置 ----------------------------------------------------*/
    RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOD, ENABLE);
    GPIO_PinAFConfig(GPIOD, GPIO_PinSource8, GPIO_AF_USART3);
    GPIO_PinAFConfig(GPIOD, GPIO_PinSource9, GPIO_AF_USART3);
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8 ;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
    GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;    
    GPIO_Init(GPIOD, &GPIO_InitStructure);  
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
    GPIO_Init(GPIOD, &GPIO_InitStructure);  
}

/*******************************************************************************
 * 函数名称:DMA_Conf
 * 功能描述: 串口DMA配置(DMA1_Channel4_Stream1->USART3_RX,
 *                       DMA1_Channel4_Stream3->USART3_TX)
 * 输入参数:none
 * 返 回 值:none
 * 作    者:roger.luo
 ******************************************************************************/
#if TTY_USE_DMA == 1
static void DMA_Conf(void)
{
    DMA_InitTypeDef DMA_Structure;
    NVIC_InitTypeDef NVIC_InitStructure;

    /* Enable DMA clock */
    RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA1, ENABLE);    
    DMA_DeInit(DMA1_Stream1);
    DMA_DeInit(DMA1_Stream3);
    while (DMA_GetCmdStatus(DMA1_Stream1) != DISABLE){}
    while (DMA_GetCmdStatus(DMA1_Stream3) != DISABLE){}
    /*配置串口3接收流 */
    DMA_Structure.DMA_Channel = DMA_Channel_4;                    /*DMA1通道4*/
    DMA_Structure.DMA_PeripheralBaseAddr = (uint32_t)(&USART3->DR);
    DMA_Structure.DMA_Memory0BaseAddr = (uint32_t)dma_rx_buf;
    DMA_Structure.DMA_DIR = DMA_DIR_PeripheralToMemory;           /*外设到内存*/
    DMA_Structure.DMA_BufferSize = sizeof(dma_rx_buf);
    DMA_Structure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
    DMA_Structure.DMA_MemoryInc = DMA_MemoryInc_Enable;
    DMA_Structure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
    DMA_Structure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
    DMA_Structure.DMA_Mode = DMA_Mode_Circular;                   /*循环模式*/
    DMA_Structure.DMA_Priority = DMA_Priority_Low;
    DMA_Structure.DMA_FIFOMode = DMA_FIFOMode_Disable;         
    DMA_Structure.DMA_FIFOThreshold = DMA_FIFOThreshold_Full;
    DMA_Structure.DMA_MemoryBurst = DMA_MemoryBurst_Single;
    DMA_Structure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
    DMA_Init(DMA1_Stream1, &DMA_Structure); 

    /*配置串口3发送流 */
    DMA_Structure.DMA_PeripheralBaseAddr = (uint32_t)(&USART3->DR);
    DMA_Structure.DMA_Memory0BaseAddr = (uint32_t)dma_tx_buf;
    DMA_Structure.DMA_DIR = DMA_DIR_MemoryToPeripheral;            /*内存到外设*/
    DMA_Structure.DMA_BufferSize = sizeof(dma_tx_buf);
    DMA_Structure.DMA_Mode = DMA_Mode_Normal;                      /*正常模式 -*/
    DMA_Init(DMA1_Stream3, &DMA_Structure); 

    /* Enable DMA Stream Transfer Complete interrupt */
    DMA_ITConfig(DMA1_Stream1, DMA_IT_TC, ENABLE);    
    //DMA_ITConfig(DMA1_Stream3, DMA_IT_TC, ENABLE);
    /* DMA Stream enable */
    DMA_Cmd(DMA1_Stream1, ENABLE);                                 /*使能接收流*/

    /* Enable the DMA Stream IRQ Channel */
    NVIC_InitStructure.NVIC_IRQChannel = DMA1_Stream1_IRQn;
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    NVIC_Init(&NVIC_InitStructure);  
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
    NVIC_InitStructure.NVIC_IRQChannel = DMA1_Stream3_IRQn;
    NVIC_Init(&NVIC_InitStructure);  
}
#endif
/*******************************************************************************
 * 函数名称:uart_conf
 * 功能描述:TTY 串口配置
 * 输入参数:none
 * 返 回 值:none
 * 作    者:roger.luo
 ******************************************************************************/
static void uart_conf(void)
{
    USART_InitTypeDef USART_InitStructure;
    NVIC_InitTypeDef NVIC_InitStructure;
    USART_DeInit(USART3);
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART3, ENABLE);
    USART_InitStructure.USART_BaudRate = TTY_BAUDRATE;  
    USART_InitStructure.USART_WordLength = USART_WordLength_8b;
    USART_InitStructure.USART_StopBits = USART_StopBits_1;
    USART_InitStructure.USART_Parity = USART_Parity_No;
    USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
    USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
    USART_Init(USART3, &USART_InitStructure);

    NVIC_InitStructure.NVIC_IRQChannel = USART3_IRQn;
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    NVIC_Init(&NVIC_InitStructure); 

    ring_buf_create(&ringbuf_send, txbuf, sizeof(txbuf));/*初始化环形缓冲区 --*/
    ring_buf_create(&ringbuf_recv, rxbuf, sizeof(rxbuf));     

#if TTY_USE_DMA == 1     
    USART_DMACmd(USART3,USART_DMAReq_Rx,ENABLE);         /*开启DMA请求 --------*/
    USART_DMACmd(USART3,USART_DMAReq_Tx,ENABLE);   
    USART_ITConfig(USART3, USART_IT_IDLE, ENABLE);       /*打开空闲中断处理DMA接收 -------*/     
#else
    USART_ITConfig(USART3, USART_IT_RXNE, ENABLE);
#endif    
    USART_ITConfig(USART3, USART_IT_ERR, ENABLE);    
    USART_Cmd(USART3, ENABLE);    
}

/*******************************************************************************
 * 函数名称:init
 * 功能描述:打印驱动初始化
 * 输入参数:none
 * 返 回 值:none
 * 作    者:roger.luo
 ******************************************************************************/
static void init(void)
{
    port_conf();
    uart_conf();
#if TTY_USE_DMA == 1    
    DMA_Conf(); 
#endif    
}


/*******************************************************************************
 * 函数名称:send
 * 功能描述:向串口发送缓冲区内写入数据
 * 输入参数:buf       -  缓冲区
 *           len       -  缓冲区长度
 * 返 回 值:实际写入长度(如果此时缓冲区满,则返回len)
 * 作    者:roger.luo
 ******************************************************************************/
static unsigned int send(void *buf, unsigned int len)
{

#if TTY_USE_DMA == 1    
    unsigned int ret;
    ret = ring_buf_put(&ringbuf_send, buf, len);  
    USART_ITConfig(USART3, USART_IT_TC, ENABLE);  
    return ret;
#else
    unsigned int ret;
    ret = ring_buf_put(&ringbuf_send, (unsigned char *)buf, len);       
    USART_ITConfig(USART3, USART_IT_TXE, ENABLE);    
    return ret;        
#endif    
}

/*******************************************************************************
 * 函数名称:recv
 * 功能描述:读取tty接收缓冲区的数据
 * 输入参数:buf       -  缓冲区
 *           len       -  缓冲区长度
 * 返 回 值:(实际读取长度)如果接收缓冲区的有效数据大于len则返回len否则返回缓冲
 *            区有效数据的长度
 * 作    者:roger.luo
 ******************************************************************************/
unsigned int recv(void *buf, unsigned int len)
{
    return ring_buf_get(&ringbuf_recv, (unsigned char *)buf, len);
}

#if TTY_USE_DMA == 1
/*******************************************************************************
 * 函数名称:DMA1_Stream1_IRQHandler
 * 功能描述:TTY串口DMA接收完成中断
 * 输入参数:none
 * 返 回 值:none
 ******************************************************************************/
void DMA1_Stream1_IRQHandler(void)
{
    if (DMA_GetITStatus(DMA1_Stream1, DMA_IT_TCIF1) != RESET)  
    { 
        ring_buf_put(&ringbuf_recv, dma_rx_buf, sizeof(dma_rx_buf));
        DMA_ClearITPendingBit(DMA1_Stream1, DMA_IT_TCIF1);
    }    
}
/*******************************************************************************
 * 函数名称:DMA1_Stream3_IRQHandler
 * 功能描述:TTY串口DMA发送完成中断
 * 输入参数:none
 * 返 回 值:none
 ******************************************************************************/
/*void DMA1_Stream3_IRQHandler(void)
{
    unsigned int len;
    if (DMA_GetITStatus(DMA1_Stream3, DMA_IT_TCIF3) != RESET)
    {
        DMA_ClearITPendingBit(DMA1_Stream3, DMA_IT_TCIF3);  
        if ((len = ring_buf_get(&ringbuf_send, dma_tx_buf, sizeof(dma_tx_buf))))
        {                
            DMA_SetCurrDataCounter(DMA1_Stream3, len);        
            DMA_Cmd(DMA1_Stream3, ENABLE);
            USART_DMACmd(USART3,USART_DMAReq_Tx,ENABLE);
        }
        else sending = 0;
    }   

}*/
#endif

/*******************************************************************************
 * 函数名称:USART3_IRQHandler
 * 功能描述:串口1收发中断
 * 输入参数:none
 * 返 回 值:none
 ******************************************************************************/
void USART3_IRQHandler(void)
{    
#if TTY_USE_DMA == 1    
    uint16_t len, retry = 0;
    if (USART_GetITStatus(USART3, USART_IT_IDLE) != RESET || 
        USART_GetITStatus(USART3, USART_IT_FE) != RESET) 
    {
        /*获取DMA缓冲区内的有效数据长度 --------------------------------------*/
        len = sizeof(dma_rx_buf) - DMA_GetCurrDataCounter(DMA1_Stream1);    
        DMA_Cmd(DMA1_Stream1, DISABLE);  
        ring_buf_put(&ringbuf_recv,dma_rx_buf, len);    /*将数据放入接收缓冲区*/   
        while (DMA_GetCmdStatus(DMA1_Stream1) != DISABLE && retry++ < 100){}  
        /*复位DMA当前计数器值 ------------------------------------------------*/
        DMA_SetCurrDataCounter(DMA1_Stream1, sizeof(dma_rx_buf));
        DMA_Cmd(DMA1_Stream1, ENABLE);         
        DMA_ClearFlag(DMA1_Stream1, DMA_FLAG_TCIF1);    /*清除传输完成标志,否则会进入传输完成中断*/
        len = USART3->DR;                               /*清除中断标志 -------*/
    }
    if (USART_GetITStatus(USART3, USART_IT_TC) != RESET)
    {  
        if ((len = ring_buf_get(&ringbuf_send, dma_tx_buf, sizeof(dma_tx_buf))))
        {                
            DMA_Cmd(DMA1_Stream3, DISABLE);
            DMA_ClearFlag(DMA1_Stream3, DMA_FLAG_TCIF3);
            DMA_SetCurrDataCounter(DMA1_Stream3, len);        
            DMA_Cmd(DMA1_Stream3, ENABLE);
        }
        else 
        {
            USART_ITConfig(USART3, USART_IT_TC, DISABLE);       
        }
    }
#else
    unsigned char data;
    if (USART_GetITStatus(USART3, USART_IT_RXNE) != RESET) 
    {
        data = USART_ReceiveData(USART3);
        ring_buf_put(&ringbuf_recv,&data, 1);           /*将数据放入接收缓冲区*/  
    }
    if (USART_GetITStatus(USART3, USART_IT_TXE) != RESET) 
    {
        if (ring_buf_get(&ringbuf_send, &data, 1))      /*从缓冲区中取出数据---*/
        {
            USART_SendData(USART3, data);            
        }
        else
            USART_ITConfig(USART3, USART_IT_TXE, DISABLE);    
    }
    if (USART_GetITStatus(USART3, USART_IT_FE) != RESET)
    {
        data = USART_ReceiveData(USART3);

    }    
#endif
}


/*******************************************************************************
 * 函数名称:putstr
 * 功能描述:输出一个字符串
 * 输入参数:none
 * 返 回 值:none
 ******************************************************************************/
static void putstr(const char *str)
{
    send((void *)str, strlen(str));
}

/*******************************************************************************
 * 函数名称:clear
 * 功能描述:清除接收缓冲区的数据
 * 输入参数:none
 * 返 回 值:none
 ******************************************************************************/
static void clear(void)
{
    ring_buf_clr(&ringbuf_recv);
}

/*******************************************************************************
 * 函数名称:buflen
 * 功能描述:清除接收缓冲区的数据
 * 输入参数:none
 * 返 回 值:none
 ******************************************************************************/
static unsigned int buflen(void)
{
    return ring_buf_len(&ringbuf_recv);
}

/*******************************************************************************
 * 函数名称:print
 * 功能描述:格式化打印输出
 * 输入参数:none
 * 返 回 值:none
 ******************************************************************************/
static void print(const char *format, ...)
{
    va_list args;
    char buf[256];
    va_start (args, format);
    vsprintf (buf, format, args);       
    va_end (args);    
    putstr(buf);

}


/*外部接口定义 --------------------------------------------------------------*/
extern const tty_t tty = 
{
    init,
    send,
    recv,
    putstr,
    clear,
    buflen,
    print
};
--------------------- 
作者:魔罗 
来源:CSDN 
原文:https://blog.csdn.net/jieffantfyan/article/details/52675299 
版权声明:本文为博主原创文章,转载请附上博文链接!

这篇关于一个免锁环形缓冲区的实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1131379

相关文章

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略 1. 特权模式限制2. 宿主机资源隔离3. 用户和组管理4. 权限提升控制5. SELinux配置 💖The Begin💖点点关注,收藏不迷路💖 Kubernetes的PodSecurityPolicy(PSP)是一个关键的安全特性,它在Pod创建之前实施安全策略,确保P

工厂ERP管理系统实现源码(JAVA)

工厂进销存管理系统是一个集采购管理、仓库管理、生产管理和销售管理于一体的综合解决方案。该系统旨在帮助企业优化流程、提高效率、降低成本,并实时掌握各环节的运营状况。 在采购管理方面,系统能够处理采购订单、供应商管理和采购入库等流程,确保采购过程的透明和高效。仓库管理方面,实现库存的精准管理,包括入库、出库、盘点等操作,确保库存数据的准确性和实时性。 生产管理模块则涵盖了生产计划制定、物料需求计划、

C++——stack、queue的实现及deque的介绍

目录 1.stack与queue的实现 1.1stack的实现  1.2 queue的实现 2.重温vector、list、stack、queue的介绍 2.1 STL标准库中stack和queue的底层结构  3.deque的简单介绍 3.1为什么选择deque作为stack和queue的底层默认容器  3.2 STL中对stack与queue的模拟实现 ①stack模拟实现

基于51单片机的自动转向修复系统的设计与实现

文章目录 前言资料获取设计介绍功能介绍设计清单具体实现截图参考文献设计获取 前言 💗博主介绍:✌全网粉丝10W+,CSDN特邀作者、博客专家、CSDN新星计划导师,一名热衷于单片机技术探索与分享的博主、专注于 精通51/STM32/MSP430/AVR等单片机设计 主要对象是咱们电子相关专业的大学生,希望您们都共创辉煌!✌💗 👇🏻 精彩专栏 推荐订阅👇🏻 单片机