单片机延时函数怎么写规范?

news/2025/2/25 19:42:03

我们以前在开发产品的时候,肯定会碰到一些延时需求,比如常见的LED闪烁,按键消抖,控制IO口输出时序等等。

别小看延时,这个小问题,想做好,甚至要考虑到程序架构层面。

在开发板上,可能你用delay死延时,很简单。

但是有个致命的问题,就是CPU阻塞,需要等延时完,程序才能往下执行,这种在实际产品大部分情况是不能用的,还有就是这种延时时间精度也不够,可能你延时500ms,实测550ms~600ms随机跳动。

如果换个主频从12MHz改为24MHz的单片机,所有定时全乱了套,改到你抓狂。

后面工作了,我就通过定时器,以全局变量来计时,然后判断变量值来判断时间,时间精度的问题解决了,但是又伴随着另一个问题,就是代码可扩展性和可移植性差,换一个项目,要增加新的延时时间,或者换一个单片机,代码又要大改。

今天带你彻底解决这个问题,分享我以前做产品一直在用的定时架构,已经经过几十个项目批量验证,稳定、可扩展,可移植。

一、架构实现思路图解

1.1 核心数据结构体

typedef struct {
    uint16_t Period;        // 定时周期(50μs单位)
    uint16_t CurrentCount;  // 当前计数值
    void (*func)(void);     // 回调函数指针
    TIMER_STATE_TYPEDEF state; // 状态标示
} Stu_TimerTypedef;

volatile Stu_TimerTypedef Stu_Timer[T_SUM]; // T_SUM建议定义8

1.2 三层架构设计

二、代码逐行解析(核心函数)

2.1 硬件初始化函数

static void hal_timer4Config(void)
{
    // TIM4时钟使能
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4, ENABLE);
    
    TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure = {0};
    TIM_TimeBaseStructure.TIM_Period = 50 - 1;  // 50us间隔自动重装载值
    TIM_TimeBaseStructure.TIM_Prescaler = SystemCoreClock/1000000 - 1; // 1MHz时基
    //其它初始化代码
}

2.2 定时器管理API

2.2.1 创建定时器

void hal_CreatTimer(TIMER_ID_TYPEDEF id, void (*proc)(void), 
                   uint16_t Period, TIMER_STATE_TYPEDEF state)
{
    Stu_Timer[id].state = state;
    Stu_Timer[id].Period = Period;   // 设置周期(50μs*Period)
    Stu_Timer[id].CurrentCount = 0;  // 清空计数
    Stu_Timer[id].func = proc;       // 绑定回调函数
}

3.2.2 定时器状态控制

TIMER_RESULT_TYPEDEF hal_CtrlTimerAction(TIMER_ID_TYPEDEF id, 
                                       TIMER_STATE_TYPEDEF sta)
{
    if(Stu_Timer[id].func != NULL){
        Stu_Timer[id].state = sta; // 修改运行状态
        return T_SUCCESS;
    }
    return T_FAIL; // 定时器未创建
}

3.3 中断处理核心

void TIM4_IRQHandler(void)
{
    if(TIM_GetITStatus(TIM4, TIM_IT_Update) != RESET){
        // 全局中断处理函数
        for(uint8_t i=0; i<T_SUM; i++){
            if(Stu_Timer[i].state == T_STA_START){ 
                if(++Stu_Timer[i].CurrentCount >= Stu_Timer[i].Period){
                    Stu_Timer[i].state = T_STA_STOP; // 单次触发模式
                    Stu_Timer[i].func(); // 执行用户回调
                }
            }
        }
        TIM_ClearITPendingBit(TIM4, TIM_IT_Update);
    }
}

三、基础用法示例

3.1 LED闪烁(1Hz)

// 定义LED任务ID
#define LED_TASK_ID 0

// LED回调函数
void LED_Task(void){
    GPIO_WriteBit(GPIOC, GPIO_Pin_13, 
                (BitAction)(1 - GPIO_ReadOutputDataBit(GPIOC, GPIO_Pin_13)));
}

int main(void){
    // 硬件初始化
    hal_timerInit();  
    GPIO_Init(GPIOC, GPIO_Pin_13, GPIO_Mode_Out_PP);
    
    // 创建定时器(10000*50μs=500ms)
    hal_CreatTimer(LED_TASK_ID, LED_Task, 10000, T_STA_START); 
    
    while(1){
        // 主循环可添加其他任务
        if(需要重启定时器){
            hal_ResetTimer(LED_TASK_ID, T_STA_START);
        }
    }
}

3.2 按键消抖(进阶用法)

#define KEY_TASK_ID 1
uint8_t key_state = 0;

void Key_Scan_Task(void){
    static uint16_t press_time = 0;
    
    if(GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_0)){
        if(++press_time > 10){ // 50μs*10=0.5ms
            key_state = 1;
        }
    }else{
        press_time = 0;
        key_state = 0;
    }
}

void Init_Key_Scan(void){
    hal_CreatTimer(KEY_TASK_ID, Key_Scan_Task, 10, T_STA_START); // 每0.5ms扫描
}

关于这个定时器架构,我在2018年也录了一套比较系统的教程,可滴滴我安排。

以上两种是比较常用了,除了这个,我们无际单片机项目里还有控制单口时序驱动外围芯片的用法,比如语音芯片等等,用起来极其灵活。

这种是通过定时器的精准定时,定时任务在定时器中断里面执行,也是有缺点的,如果定时的任务多了,就会影响实时性。

所以,有些定时,不需要要求这么高的,我们一般是配合任务的Tick,然后每个任务里设置一个变量,通过递增和递减来延时。

之前有同学问过我,怎么去验证这个定时器时间准不准?

我们在调试延时架构代码的阶段,会通过示波器,配合IO电平翻转去测试,比如10ms翻转一次,看下精度。


最近很多粉丝问我单片机怎么学,我根据自己从业十年经验,累积耗时一个月,精心整理一份「单

片机最佳学习路径+单片机入门到高级教程+工具包」全部无偿分享给铁粉!!!

除此以外,再含泪分享我压箱底的22个热门开源项目,包含源码+原理图+PCB+说明文档,让你迅速进阶成高手

教程资料包和详细的学习路径可以看我下面这篇文章的开头

单片机入门到高级开挂学习路径(附教程+工具)》

单片机入门到高级开挂学习路径(附教程+工具)》

单片机入门到高级开挂学习路径(附教程+工具)》


http://www.niftyadmin.cn/n/5865895.html

相关文章

《数据库索引设计与优化》译本错误纠正(1)

今天在学习《数据库索引设计与优化》第十一章第198页的时候遇到一个问题&#xff0c;即参数的文字描述与实际不符。我看的是从网络上找到的译本&#xff0c;许多喜欢白嫖的朋友可能也会像我一样遇到这种问题。 可以看到&#xff0c;上面对参数Z的描述是&#xff1a;Z上一次索引…

快速排序、大根堆排序比较,C++详解

目录 实验题目 问题分析 快速排序 大根堆排序 两种排序方法时间复杂度的比较 总的代码 运行结果 实验题目 实验题目: 对快速排序、大根堆排序&#xff0c;两种排序方法进行比较&#xff0c;写出大根堆排序算法。并对它们最坏、最好&#xff0c;平均情况进行比较…

redis主从哨兵模式+Lua报错-READONLY You can‘t write against a read

背景 项目试用SpringBootredisTemplate执行redis的lua脚本&#xff0c;实现令牌桶&#xff1b;redis结构使用的是1主2从3哨兵模式读写分离&#xff1b; 问题分析 READONLY You cant write against a read报这个错的含义在从节点执行了写操作&#xff0c;也就是说我执行Lua脚…

AI手机的技术细节

前序&#xff1a;先说各个功能涉及到的技术&#xff0c;再说宏观系统架构。AI手机有这样几个做法&#xff0c;给手机侧边增加一个按键&#xff1b;把手机的语音助手做的很好&#xff0c;能够快速稳定的进行唤醒&#xff1b;通过特殊形式的触摸手机的曲面屏位置等来进行唤醒AI …

信息学奥赛一本通 1522:网络 | OpenJudge 百练 1144:Network

【题目链接】 ybt 1522&#xff1a;网络 OpenJudge 百练 1144:Network 【题目考点】 1. 图论&#xff1a;割点 【解题思路】 每个交换机是一个顶点&#xff0c;如果两地点之间有电话线连接&#xff0c;那么两顶点之间有一条无向边&#xff0c;该图是无向图。 初始时任何地…

TD时间差分算法

TD算法用来估计value-state 给定data/experiece of algorithm&#xff0c; TD算法&#xff1a; 其中TD error&#xff1a; δ t v ( s t ) − [ r t 1 γ v ( s t 1 ) ] v ( s t ) − v t ‾ \delta_t v(s_t) -[r_{t1} \gamma v(s_{t1})]v(s_t) - \overline{v_{t}} δ…

Vue使用Three.js加载glb (gltf) 文件模型及实现简单的选中高亮、测距、测面积

安装&#xff1a; # three.jsnpm install --save three 附中文网&#xff1a; 5. gltf不同文件形式(.glb) | Three.js中文网 附官网&#xff1a; 安装 – three.js docs 完整代码&#xff08;简易demo&#xff09;&#xff1a; <template><div class"siteInspe…

【复习】计算机网络

网络模型 OSI 应用层&#xff1a;给应用程序提供统一的接口表示层&#xff1a;把数据转换成兼容另一个系统能识别的格式会话层&#xff1a;负责建立、管理、终止表示层实体之间的通信会话传输层&#xff1a;负责端到端的数据传输网络层&#xff1a;负责数据的路由、转发、分片…