一、单片机延时分类 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的资源,无法执行其他任务,而非阻塞式延时可以让程序在执行延时函数时继续执行其他任务,提高了程序的效率。但是非阻塞式延时需要使用定时器或者其他硬件资源来实现,实现起来相对复杂一些。
|