STM32延时函数的三种方法——最好掌握第三种

单片机编程过程中经常用到延时函数,最常用的莫过于微秒级延时delay_us( )和毫秒级delay_ms( )。

1.普通延时法

这个比较简单,让单片机做一些无关紧要的工作来打发时间,经常用循环来实现,不过要做的比较精准还是要下一番功夫。下面的代码是在网上搜到的,经测试延时比较精准。

//粗延时函数,微秒

void delay_us(u16 time)
{    
   u16 i=0;  
   while(time--)
   {
      i=10;  //自己定义
      while(i--) ;    
   }
}
//毫秒级的延时
void delay_ms(u16 time)
{    
   u16 i=0;  
   while(time--)
   {
      i=12000;  //自己定义
      while(i--) ;    
   }
}

2.SysTick 定时器延时

CM3 内核的处理器,内部包含了一个SysTick 定时器,SysTick 是一个24 位的倒计数定时器,当计到0 时,将从RELOAD寄存器中自动重装载定时初值。只要不把它在SysTick 控制及状态寄存器中的使能位清除,就永不停息。SysTick 在STM32的参考手册里面介绍的很简单,其详细介绍,请参阅《Cortex-M3 权威指南》。

 这里面也有两种方式实现:

a.中断方式 如下,定义延时时间time_delay,SysTick_Config()定义中断时间段,在中断中递减time_delay,从而实现延时。

volatile unsigned long time_delay; // 延时时间,注意定义为全局变量
//延时n_ms
void delay_ms(volatile unsigned long nms)
{
    //SYSTICK分频--1ms的系统时钟中断
    if (SysTick_Config(SystemFrequency/1000))
    {
   
        while (1);
    }
    time_delay=nms;//读取定时时间
    while(time_delay);
    SysTick->CTRL=0x00; //关闭计数器
    SysTick->VAL =0X00; //清空计数器
}
//延时nus
void delay_us(volatile unsigned long nus)
{
 //SYSTICK分频--1us的系统时钟中断
    if (SysTick_Config(SystemFrequency/1000000))
    {
   
        while (1);
    }
    time_delay=nus;//读取定时时间
    while(time_delay);
    SysTick->CTRL=0x00; //关闭计数器
    SysTick->VAL =0X00; //清空计数器
}

    //在中断中将time_delay递减。实现延时

void SysTick_Handler(void)
{
    if(time_delay)
        time_delay--;
}

b.非中断方式

主要仿照原子的《STM32不完全手册》。SYSTICK 的时钟固定为HCLK 时钟的1/8,在这里我们选用内部时钟源72M,所以SYSTICK的时钟为9M,即SYSTICK定时器以9M的频率递减。SysTick 主要包含CTRL、LOAD、VAL、CALIB 等4 个寄存器,

SysTick->CTRL

位段

名称

类型

复位值

描述

16

COUNTFLAG

R

0

如果在上次读本寄存器后systick已为0,则该位为1,若 读该位自动清零

2

CLKSOURCE

RW

0

0:外部时钟源 1:内部时钟

1

TICKINT

RW

0

0:减到0无动作;1:减到0产生systick异常请求

0

ENABLE

RW

0

systick定时器使能位

    

 

SysTick-> LOAD

位段

名称

类型

复位值

描述

23:0

RELOAD

RW

0

减到0时被重新装载的值

SysTick-> VAL

位段

名称

类型

复位值

描述

23:0

CURRENT

RW

0

读取时返回当前倒计数的值,写则清零,同时还会清除在systick控制及状态寄存器中的COUNTFLAG标志

SysTick-> CALIB 不常用,在这里我们也用不到,故不介绍了。

程序如下,相当于查询法。

//仿原子延时,不进入systic中断
void delay_us(u32 nus)
{
 u32 temp;
 SysTick->LOAD = 9*nus;
 SysTick->VAL=0X00;//清空计数器
 SysTick->CTRL=0X01;//使能,减到零是无动作,采用外部时钟源
 do
 {
  temp=SysTick->CTRL;//读取当前倒计数值
 }while((temp&0x01)&&(!(temp&(1<<16))));//等待时间到达
     SysTick->CTRL=0x00; //关闭计数器
    SysTick->VAL =0X00; //清空计数器
}
void delay_ms(u16 nms)
{
 u32 temp;
 SysTick->LOAD = 9000*nms;
 SysTick->VAL=0X00;//清空计数器
 SysTick->CTRL=0X01;//使能,减到零是无动作,采用外部时钟源
 do
 {
  temp=SysTick->CTRL;//读取当前倒计数值
 }while((temp&0x01)&&(!(temp&(1<<16))));//等待时间到达
    SysTick->CTRL=0x00; //关闭计数器
    SysTick->VAL =0X00; //清空计数器
}

三种方式各有利弊,第一种方式容易理解,但不太精准。第二种方式采用库函数,编写简单,由于中断的存在,不利于在其他中断中调用此延时函数。第三种方式直接操作寄存器,看起来比较繁琐,其实也不难,同时克服了以上两种方式的缺点,个人感觉比较好用。

函数和中断服务函数,程序框架如下: ``` #include <reg52.h> #define uchar unsigned char #define uint unsigned int sbit key1 = P3^1; // 按键1 sbit key2 = P3^2; // 按键2 sbit key3 = P3^3; // 按键3 uchar hour = 0; // 小时 uchar minute = 0; // 分钟 uchar second = 0; // 秒钟 uchar alarm_hour1 = 0; // 闹钟1小时 uchar alarm_minute1 = 0; // 闹钟1分钟 uchar alarm_second1 = 0; // 闹钟1秒钟 uchar alarm_hour2 = 0; // 闹钟2小时 uchar alarm_minute2 = 0; // 闹钟2分钟 uchar alarm_second2 = 0; // 闹钟2秒钟 uchar alarm_hour3 = 0; // 闹钟3小时 uchar alarm_minute3 = 0; // 闹钟3分钟 uchar alarm_second3 = 0; // 闹钟3秒钟 // 数码管段码表 uchar code table[] = {0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f}; // 数码管位选表 uchar code place[] = {0xfe,0xfd,0xfb,0xf7,0xef,0xdf}; // 数码管显示函数 void display(uchar hour, uchar minute, uchar second) { uchar digit[6]; digit[0] = hour / 10; digit[1] = hour % 10; digit[2] = minute / 10; digit[3] = minute % 10; digit[4] = second / 10; digit[5] = second % 10; for (uchar i = 0; i < 6; i++) { P2 = table[digit[i]]; P0 = place[i]; delay(1); // 延时1ms P0 = 0xff; // 关闭数码管 } } // 计时器中断服务函数 void timer0() interrupt 1 { TH0 = 0xfc; TL0 = 0x18; second--; if (second == 0xff) { second = 59; minute--; if (minute == 0xff) { minute = 59; hour--; if (hour == 0xff) { hour = 23; } } } display(hour, minute, second); } // 延时函数 void delay(uint ms) { uint i, j; for (i = ms; i > 0; i--) { for (j = 110; j > 0; j--); } } // 主函数 void main() { TMOD = 0x01; // 定时器0工作模式1 TH0 = 0xfc; TL0 = 0x18; ET0 = 1; // 允许定时器0中断 EA = 1; // 允许中断总开关 TR0 = 1; // 启动定时器0 while (1) { if (!key1) { // 按键1设置闹钟1 delay(10); if (!key1) { alarm_hour1 = hour; alarm_minute1 = minute; alarm_second1 = second; } while (!key1); } if (!key2) { // 按键2设置闹钟2 delay(10); if (!key2) { alarm_hour2 = hour; alarm_minute2 = minute; alarm_second2 = second; } while (!key2); } if (!key3) { // 按键3设置闹钟3 delay(10); if (!key3) { alarm_hour3 = hour; alarm_minute3 = minute; alarm_second3 = second; } while (!key3); } if (hour == alarm_hour1 && minute == alarm_minute1 && second == alarm_second1) { alarm(); alarm_hour1 = 0; alarm_minute1 = 0; alarm_second1 = 0; } if (hour == alarm_hour2 && minute == alarm_minute2 && second == alarm_second2) { alarm(); alarm_hour2 = 0; alarm_minute2 = 0; alarm_second2 = 0; } if (hour == alarm_hour3 && minute == alarm_minute3 && second == alarm_second3) { alarm(); alarm_hour3 = 0; alarm_minute3 = 0; alarm_second3 = 0; } } } // 警报函数 void alarm() { P2 = 0xff; P0 = 0x00; delay(2000); P0 = 0xff; } ``` 注意:以上代码仅供参考,具体实现还需要根据硬件电路和需求进行调整和完善。
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值