STM32F103入门 | 8.精准延时

工程模板的建立点亮灯小实验中,我们可以发现其中都用到了延时函数Delay。比如点亮小灯那里: Delay(900000); 相信很多小伙伴刚开始看到这句的时候会觉得:哇~好大的数字,这是要等到花儿都谢了的节奏吗!其实,把代码下载到开发板观察LED闪烁的时候,它的闪烁频率还是非常快的......

诚然,我们需要一个比较准一点的延时方案。这个延时功能既要实在,又得优雅。既然这样,我们可以写一个毫秒级延时和一个微秒级延时。下面分享精准延时方案。

 

1. 新建两个文件,delay.c 和 delay.h

 

 

 

 

2. 在头文件 delay.h 添加下面代码:

#ifndef _DELAY_H
#define _DELAY_H
#include "stm32f10x.h"

void delay_init(void);
void delay_ms(u16 nms);
void delay_us(u32 nus);

#endif

 

3. 把 delay.c 添加到工程中,很多人都会忘记这一步,代码写好了却发现在工程那里看不到。现在文件不多,我就把它添加到USER目录下吧。这个怎么喜欢怎么来。但是,但是,养成良好的编程习惯会让自己走得更远哦。后面随着文件的增多和代码量的增大,我们就需要进行分类和归档啦。

 

 

4. 在 delay.c 中添加以下代码:

#include “delay.h”

static u8  fac_us=0; //us延时倍乘数
static u16 fac_ms=0; //ms延时倍乘数

void delay_init(void)
{
    SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);    //选择外部时钟 HCLK/8
    fac_us = SystemCoreClock/8000000;                        //为系统时钟的1/8
    fac_ms = (u16)fac_us*1000;                               //每个ms需要的systick时钟数
}

void delay_us(u32 nus)
{
    u32 temp;
    SysTick->LOAD = nus*fac_us;                 //时间加载
    SysTick->VAL = 0x00;                        //清空计数器
    SysTick->CTRL |= SysTick_CTRL_ENABLE_Msk ;  //开始倒数
    do
    {
        temp = SysTick->CTRL;
    }while((temp&0x01)&&!(temp&(1<<16)));       //等待时间到达
    SysTick->CTRL &= ~SysTick_CTRL_ENABLE_Msk;  //关闭计数器
    SysTick->VAL = 0x00;                        //清空计数器
}

void delay_ms(u16 nms)
{
    u32 temp;
    SysTick->LOAD = (u32)nms*fac_ms;            //时间加载(SysTick->LOAD为24bit)
    SysTick->VAL = 0x00;                        //清空计数器
    SysTick->CTRL |= SysTick_CTRL_ENABLE_Msk ;  //开始倒数
    do
    {
        temp=SysTick->CTRL;
    }while((temp&0x01)&&!(temp&(1<<16)));       //等待时间到达
    SysTick->CTRL &= ~SysTick_CTRL_ENABLE_Msk;  //关闭计数器
    SysTick->VAL = 0x00;                        //清空计数器
}

这里啰嗦一下,delay 是不可重入的,也就是说,如果在其他地方执行 delay 延时的时候,出现了中断更新事件,这时如果中断服务函数中也用到了 delay 函数来延时,那么就会发生延时不准确的情况。这时,想要更优的方案就是在中断中使用软延时,让它数数。至于怎么数才能准确,这个后面会提及。另外,delay_ms() 函数中,对72MHz条件下,nms<=1864。如果想延时长一点时间,写多几句 delay_ms() 不就完事了~

 

5. 把 delay 用起来。我就用点亮小灯的代码吧,ctrl+c、ctrl+v 、改几下,多简单。如下图:

 

算了,再啰嗦一下......这个点亮LED灯的功能后面还会经常使用。(ctrl+c、ctrl+v 、改几下)有空还是把它写到两个文件中吧,例如:led.c 和 led.h

编译、下载、灯闪、完成!

 

欢迎关注微信公众号『OpenSSR』

### STM32F103C8T6 图解资料教程 #### 硬件结构概述 STM32F103C8T6是一款基于ARM Cortex-M3内核的微控制器,具有丰富的外设接口。该器件集成了多个通信模块,其中包括三个USART端口用于串行数据传输[^1]。 对于希望深入了解此款MCU硬件架构的人士而言,推荐查阅官方提供的《STM32F10xxx参考手册》,其中包含了详细的框图以及各功能单元的工作原理说明。此外,《STM32CubeMX用户指南》也提供了图形化的配置工具帮助开发者快速理解并初始化各个外设参数。 #### 外围电路设计指导 当构建围绕STM32F103C8T6的应用系统时,稳定的供电至关重要。通常会采用AMS1117这类低压差线性稳压器来为MCU提供所需电压水平[^2]。为了确保良好的电气性能,在PCB布局阶段应特别注意电源层与地平面的设计,保持信号路径短捷且远离干扰源。 #### 编程入门实例解析 针对初学者来说,通过实际项目学习编程技巧是非常有效的途径之一。例如实现超声波传感器测距功能的过程中涉及到精确延时控制,可以利用TIM2定时器完成微妙级别的等待操作[^3]: ```c void TIM2_Delay_us(uint16_t n_us) { __HAL_TIM_ENABLE(&htim2); // 开始计数,启动定时器2 __HAL_TIM_SetCounter(&htim2, 0); // 设置定时器初始值 while (__HAL_TIM_GetCounter(&htim2) < (1 * n_us - 1)); // 减1是为了必须达到我们想要的值再跳出循环 __HAL_TIM_DISABLE(&htim2); // 停止计数,停止定时器2 } ``` 上述代码片段展示了如何使用硬件定时器来进行精准的时间延迟处理,这对于驱动外部设备或执行特定时间间隔的任务非常有用。 #### 推荐参考资料获取渠道 - STMicroelectronics官方网站:这里不仅有最权威的产品文档和技术支持服务,还有众多社区论坛供爱好者交流经验心得。 - GitHub平台上的开源项目库:许多资深工程师分享了自己的作品案例,能够作为很好的实践范本进行研究模仿。 - 各类在线教育网站如慕课网、哔哩哔哩等平台上也有不少高质量的教学视频可供观看学习。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值