[技术讨论]关于单片机延时的实现讨论

一、单片机延时分类
1、软件延时:在程序中使用循环等语句来消耗一定的时间,从而实现延时,这种方式简单易用,但会占用CPU资源,影响程序的响应性能。
2、硬件延时:通过外部电路或者定时器等硬件模块来实现延时,这种方式可以不占用CPU资源,但需要占用额外的硬件资源,增加系统成本和复杂度。
3、系统延时:一些单片机芯片提供了硬件延时和软件延时的组合方式,例如使用定时器和中断相结合的方式来实现精确的延时,这种方式既能够满足延时的精度要求,又能够减少CPU的占用率,提高系统的响应性能。
4、外部时钟:通过外部时钟来实现精确的延时,这种方式需要使用外部时钟电路,并配置好单片机的时钟源,可以实现高精度的延时,但同时也增加了系统的复杂度和成本。

二、软件延时举例
下面是一个简单的单片机软件延时的演示代码,以STM32为例,使用SysTick中断来实现延时:
#include "stm32f10x.h"

void SysTick_Handler(void)

{

    static uint32_t ticks = 0;

    ticks++;    // 计数器加1

}

void delay_ms(uint32_t ms)

{

    uint32_t ticks = 0;

    ticks = ms * SystemCoreClock / 1000;    // 计算延时的时钟周期数

    SysTick->CTRL = 0;    // 关闭SysTick定时器

    SysTick->LOAD = ticks - 1;    // 设置定时器的重装载值

    SysTick->VAL = 0;    // 清空定时器的计数器

    SysTick->CTRL = 0x00000005;    // 开启SysTick定时器,使用CPU时钟源

    while(!(SysTick->CTRL & 0x00010000));    // 等待延时结束

}

int main(void)

{

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);    // 使能GPIOC时钟

    GPIO_InitTypeDef GPIO_InitStructure;

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13;

    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;

    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

    GPIO_Init(GPIOC, &GPIO_InitStructure);    // 初始化PC13引脚为推挽输出模式

    

    while(1)

    {

        GPIO_SetBits(GPIOC, GPIO_Pin_13);    // PC13输出高电平

        delay_ms(1000);    // 延时1秒

        GPIO_ResetBits(GPIOC, GPIO_Pin_13);    // PC13输出低电平

        delay_ms(1000);    // 延时1秒

    }

}

上述代码中,delay_ms()函数使用SysTick定时器实现了精确的延时,可以实现LED灯的闪烁效果。具体来说,delay_ms()函数首先计算出延时的时钟周期数,然后关闭SysTick定时器,设置定时器的重装载值,清空计数器,最后开启定时器,等待延时结束。使用SysTick定时器可以实现精确的延时,同时不会占用CPU的资源。
三、硬件延时的举例
下面是一个单片机系统硬件延时的举例,以STM32为例,使用定时器来实现延时:
#include "stm32f10x.h"

void delay_us(uint32_t us)

{

    TIM_Cmd(TIM3, DISABLE);    // 关闭TIM3定时器

    TIM_SetCounter(TIM3, 0);    // 清空计数器

    TIM_SetAutoreload(TIM3, us - 1);    // 设置重装载值

    TIM_Cmd(TIM3, ENABLE);    // 开启TIM3定时器

    while(TIM_GetFlagStatus(TIM3, TIM_FLAG_Update) == RESET);    // 等待计数器溢出

    TIM_ClearFlag(TIM3, TIM_FLAG_Update);    // 清除计数器溢出标志

}

int main(void)

{

    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);    // 使能TIM3时钟

    TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;

    TIM_TimeBaseStructure.TIM_Period = 65535;    // 设置定时器周期为65535

    TIM_TimeBaseStructure.TIM_Prescaler = 71;    // 设置预分频系数为72-1

    TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;    // 设置时钟分割为1

    TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;    // 设置计数器为向上计数模式

    TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure);    // 初始化TIM3定时器

    

    GPIO_InitTypeDef GPIO_InitStructure;

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);    // 使能GPIOC时钟

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13;

    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;

    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

    GPIO_Init(GPIOC, &GPIO_InitStructure);    // 初始化PC13引脚为推挽输出模式

    

    while(1)

    {

        GPIO_SetBits(GPIOC, GPIO_Pin_13);    // PC13输出高电平

        delay_us(1000000);    // 延时1秒

        GPIO_ResetBits(GPIOC, GPIO_Pin_13);    // PC13输出低电平

        delay_us(1000000);    // 延时1秒

    }

}

上述代码中,delay_us()函数使用TIM3定时器实现了硬件延时,可以实现LED灯的闪烁效果。具体来说,delay_us()函数首先关闭TIM3定时器,然后清空计数器,设置重装载值,开启定时器,等待计数器溢出,最后清除计数器溢出标志。使用定时器可以实现高精度的延时,同时不会占用CPU的资源。
四、阻塞式延时和非阻塞式延时的区别
阻塞式延时和非阻塞式延时是两种不同的延时方式,主要区别在于是否会阻塞程序的执行。
阻塞式延时是指程序在执行延时函数时会一直等待,直到延时结束后才会继续执行下一条指令。在延时期间,程序无法执行其他任务。例如,以下代码是一个阻塞式延时的例子:
void delay_ms(uint32_t ms)

{

    uint32_t i, j;

    for(i = 0; i < ms; i++)

    {

        for(j = 0; j < 7200; j++);    // 延时1毫秒

    }

}
在上述代码中,程序在执行延时函数时会一直等待,直到延时结束后才会继续执行下一条指令。如果需要延时1秒,程序会占用1秒的时间,无法执行其他任务。
非阻塞式延时是指程序在执行延时函数时可以继续执行其他任务,不会一直等待。在延时期间,程序可以执行其他任务。例如,以下代码是一个非阻塞式延时的例子:
void delay_ms(uint32_t ms)

{

    uint32_t ticks = ms * SystemCoreClock / 1000;

    uint32_t start = SysTick->VAL;

    while((SysTick->VAL - start) < ticks);

}
在上述代码中,程序在执行延时函数时可以继续执行其他任务,不会一直等待。如果需要延时1秒,程序可以在延时期间执行其他任务,只有在延时结束后才会继续执行延时函数中的下一条指令。
总的来说,阻塞式延时会占用CPU的资源,无法执行其他任务,而非阻塞式延时可以让程序在执行延时函数时继续执行其他任务,提高了程序的效率。但是非阻塞式延时需要使用定时器或者其他硬件资源来实现,实现起来相对复杂一些。
---------------------
作者:王小琪
链接:https://bbs.21ic.com/icview-3293776-1-1.html
来源:21ic.com
此文章已获得原创/原创奖标签,著作权归21ic所有,任何人未经允许禁止转载。

 

  • 2
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值