只需两步便可生成 51 单片机最精准的延时函数

前言#

我们在学习 51 单片机的过程中会用到延时,比如一个简单的流水灯就需要延时来控制依次点亮的时间,或者一些模块在单片机发出读数据指令后,需要延时几十微秒才可以读出数据等等,这些都离不开延时,所以我们需要一个精准的延时函数来满足我们的需求。

本篇介绍一个最简单并且延时最精准的 51 单片机延时函数的生成方法。

STC-ISP#

我们说学习 51 单片机,大部分学习的都是国产的 STC89C51 单片机,我就是从这款单片机入门的。

STC89C51 是 STC 这家公司研发生产的,同时 STC 提供了一个下载编程烧录软件——STC-ISP,这款软件可是一个好东西,不会有朋友只用它来下载程序吧?

它有好多强大且实用的功能,本篇介绍一下它的软件延时计算器功能。

下载 STC-ISP#

进入 STC 官网,将页面向下就可以找到下载链接啦。

STC 官网 >> 点击跳转

使用 STC-ISP 生成函数#

打开软件,找到“ 软件延时计算器 ”,设置参数后,点击生成代码后复制即可。

注意:设置的参数一定要和使用的单片机参数相匹配。

优化生成的代码#

生成的代码具有局限性#

void Delay1ms()		//@11.0592MHz
{
	unsigned char i, j;

	_nop_();
	i = 2;
	j = 199;
	do
	{
		while (--j);
	} while (--i);
}

上面是我从软件中生成复制的代码,软件已经自动生成了一个函数供我们调用,短短几步就做好了一个延时函数,确实不错。

但这个函数在调用时只能延时 1ms ,如果说我想延时 2ms、3ms、4ms... 难道要不停的调用函数吗?或者再去软件中生成新的延时函数?那岂不是很麻烦。

其实不必这样,我们只需简单的优化一下代码,就可以实现我们想要的功能。

一步使代码变为万能#

软件所生成的函数是延时 1ms,就是说单片机执行这个函数的程序体时用时为 1ms,那么首先我们用 while 循环把程序体框住,然后每执行一次让控制 while 循环结束的变量减一,这个变量我们通过形参传递到函数中。

注意:当使用 _nop_() 函数(可理解为软件延时)时,必须在开头添加头文件 #include <intrins.h>

_nop_() 函数相当于一个空操作(可以理解为 NOP 空操作指令),而 _nop_() 函数的空操作产生的时间与晶振有关,所以在上文中设置参数要与使用的单片机参数相匹配。

优化后的代码如下:

#include <intrins.h>

void Delay1ms(unsigned int _ms)		//@11.0592MHz
{
    unsigned char i, j;

    while (_ms--)
    {
        _nop_();
        i = 2;
        j = 199;
        do
        {
            while (--j);
        } while (--i);
    }
}

调用延时函数#

经过我们优化后的延时函数在调用时极其简单,只需在调用函数的语句中放入实参就好啦。

调用演示代码如下:

#include <reg52.h>
#include <intrins.h>

void Delay1ms(unsigned int _ms);        /* 声明延时函数 */

void main()
{
    Delay1ms(1);        /* 实参为 1,则延时 1ms */

    Delay1ms(20);       /* 实参为 20,则延时 20ms */

    Delay1ms(500);      /* 实参为 500,则延时 500ms */

    /* ...... */
}


void Delay1ms(unsigned int _ms)		//@11.0592MHz
{
    unsigned char i, j;

    while (_ms--)
    {
        _nop_();
        i = 2;
        j = 199;
        do
        {
            while (--j);
        } while (--i);
    }
}

后记#

至此,51 单片机的延时函数就编写完成啦, 快去试着生成一个延时函数,将它应用到你的项目当

1.51单片机延时,晶振为11.0592MHz

(1)粗略延时

<span style="color:#000000"><span style="background-color:#eeeeee"><code class="language-scss">
 void <span style="color:#0000ff">delay_ms</span>(uint x)
{
    uint <span style="color:#0000ff">i</span>,j;
    <span style="color:#0000ff">for</span>(i=x;i><span style="color:#880000">0</span>:i--)
        <span style="color:#0000ff">for</span>(j=<span style="color:#880000">110</span>;j><span style="color:#880000">0</span>;j--);
}
</code></span></span>

(2)定时器延时

<span style="color:#000000"><span style="background-color:#eeeeee"><code class="language-csharp">
<span style="color:#0000ff">void</span> <span style="color:#a31515">delay_ms</span>(<span style="color:#0000ff">uint</span> i)
{
    TMOD=<span style="color:#880000">0x01</span>;   <span style="color:#008000">//设置定时器工作模式</span>
    <span style="color:#0000ff">while</span>(i != <span style="color:#880000">0</span>)
    {
        TR0=<span style="color:#880000">1</span>;   <span style="color:#008000">//开启定时器</span>
        TH0=(<span style="color:#880000">65535</span><span style="color:#880000">-1000</span>)/<span style="color:#880000">256</span>;    <span style="color:#008000">//赋初值</span>
        TL0=(<span style="color:#880000">65535</span><span style="color:#880000">-1000</span>)%<span style="color:#880000">256</span>;
        <span style="color:#0000ff">while</span>(TF0 != <span style="color:#880000">1</span>);   <span style="color:#008000">//溢出标志</span>
        TF0=<span style="color:#880000">0</span>;
        i--;
    }
    TR0=<span style="color:#880000">0</span>;    <span style="color:#008000">//关闭定时器</span>
}
</code></span></span>

2.stm32l151C8T6延时,外部晶振8MHz

(1)粗略延时

<span style="color:#000000"><span style="background-color:#eeeeee"><code class="language-cpp">
<span style="color:#a31515">void</span> <span style="color:#a31515">delay_us</span>(<span style="color:#a31515">uint32_t</span> time)     <span style="color:#008000">//us延时</span>
{
    <span style="color:#a31515">uint32_t</span> i=<span style="color:#880000">4</span>\*time;
    <span style="color:#0000ff">while</span>(i--);
}
<span style="color:#a31515">void</span> <span style="color:#a31515">delay_us</span>(<span style="color:#a31515">uint32_t</span> time)     <span style="color:#008000">//ms延时</span>
{
    <span style="color:#a31515">uint32_t</span> i=<span style="color:#880000">4000</span>\*time;
    <span style="color:#0000ff">while</span>(i--);
}
</code></span></span>

(2)使用nop延时
通过使用__NOP()函数进行延时,因为使用了8M晶振4倍频,所以是32MHz,所以一个nop约等于1/32us,所以使用32个nop函数为一个us,然后根据需要的定时时间进行计算。

<span style="color:#000000"><span style="background-color:#eeeeee"><code class="language-cpp">
<span style="color:#a31515">void</span> <span style="color:#a31515">delay_us</span>(<span style="color:#a31515">uint32_t</span> time)      <span style="color:#008000">//us延时</span>
{  
    <span style="color:#a31515">uint32_t</span> i=<span style="color:#880000">0</span>;  
    <span style="color:#0000ff">for</span>(i=<span style="color:#880000">0</span>;i</code></span></span>

(3)利用SysTick延时


void delay_init()       //初始化
{
	SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);  //选择外部时钟
	fac_us=SystemCoreClock/8000000; //为系统时钟的1/8  4
	fac_ms=1000\*fac_us;
}
void delay_us(uint16_t nus)        //延时us
{
	uint32_t ui_tmp=0x00;
	SysTick->LOAD=nus\*fac_us;
	SysTick->VAL=0x00;
	SysTick->CTRL=0x01;	
	do
	{
		ui_tmp=SysTick->CTRL;
	}while((ui_tmp&0x01) && (!(ui_tmp & (1<<16))));	
	SysTick->CTRL=0x00;
	SysTick->VAL=0x00;
}
void delay_ms(uint16_t nms)         //延时ms
{
	uint32_t ui_tmp=0x00;
	SysTick->LOAD=nms\*fac_ms;
	SysTick->VAL=0x00;
	SysTick->CTRL=0x01;
	do
	{
		ui_tmp=SysTick->CTRL;
	}while((ui_tmp&0x01) && (!(ui_tmp&(1<<16))));	
	SysTick->VAL=0x00;
	SysTick->CTRL=0x00;
}
void SysTick_Handler(void)
{ 
	flag=~flag;
}

(4)定时器延时


void TIM3_Int_Init(uint16_t arr,uint16_t psc)
{
	TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
	NVIC_InitTypeDef NVIC_InitStructure;
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);	
	//设置在下一个更新事件装入活动的自动重装载寄存器周期的值,计数10000为1s;
	TIM_TimeBaseStructure.TIM_Period = arr; 	
	//设置用来作为TIMx时钟频率除数的预分频值,10kHz的计数频率
	TIM_TimeBaseStructure.TIM_Prescaler = psc;	
	//设置时钟分割:TDIS = Tck_tim
	TIM_TimeBaseStructure.TIM_ClockDivision = 0;	
	//设置TIM向上计数模式
	TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;	
	//初始化TIMx的时间基数单位
	TIM_TimeBaseInit(TIM3,&TIM_TimeBaseStructure);	
	//使能指定的TIM3中断,允许更新中断
	TIM_ITConfig(TIM3,TIM_IT_Update,ENABLE);	
	//TIM3中断
	NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn;
	//抢占优先级 0 级
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
	//从优先级 3 级
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;
	//IRQ通道被使能
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
	//初始化外设NVIC寄存器
	NVIC_Init(&NVIC_InitStructure);	
	//使能TIMx外设
	TIM_Cmd(TIM3,ENABLE);
}
void TIM3_IRQHandler(void)
{
	if(TIM_GetITStatus(TIM3,TIM_IT_Update) != RESET)  //检查指定的TIM中断发生与否
	{
		TIM_ClearITPendingBit(TIM3,TIM_IT_Update);  //清除TIMx的中断待处理位	
		if(flag==0)  
		{
			flag=1;
			GPIO_ResetBits(GPIOB,GPIO_Pin_0) ;	
		}
		else
		{
			flag=0;
			GPIO_SetBits(GPIOB,GPIO_Pin_0);
		}		
	}
}
注意:定时时间的计算
定时器时钟为:CK_CLK 
预分频数值:PSC 
自动装载寄存器数值:ARR 
进入中断的次数:time 
t=time\*(ARR+1)\*(PSC+1)/(CK_CLK)

中吧。

单片机中使用C语言实现延时函数_单片机c语言延时程序-CSDN博客  https://blog.csdn.net/DevProPlus/article/details/133283189?ops_request_misc=%257B%2522request%255Fid%2522%253A%2522C072FCD9-5A1D-4EF1-990D-8CCB667C5940%2522%252C%2522scm%2522%253A%252220140713.130102334..%2522%257D&request_id=C072FCD9-5A1D-4EF1-990D-8CCB667C5940&biz_id=0&utm_medium=distribute.pc_search_result.none-task-blog-2~all~top_click~default-1-133283189-null-null.142^v100^pc_search_result_base2&utm_term=%E5%8D%95%E7%89%87%E6%9C%BA%E5%BB%B6%E6%97%B6%E5%87%BD%E6%95%B0&spm=1018.2226.3001.4187
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值