当前位置: 首页 > news >正文

只需两步便可生成 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  4fac_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_timTIM_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

相关文章:

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

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

TETFN情感计算的实践复现(论文复现)

TETFN情感计算的实践复现&#xff08;论文复现&#xff09; 本文所涉及所有资源均在传知代码平台可获取 文章目录 TETFN情感计算的实践复现&#xff08;论文复现&#xff09;概述研究背景主要贡献模型框架数据介绍及下载复现过程&#xff08;重要&#xff09;运行过程及结果 概…...

游戏各个知识小点汇总

抗锯齿原理记录 SSAA&#xff1a;把成像的图片放大N倍&#xff0c;然后每N个点进行平均值计算。一般N为2的倍数。比如原始尺寸是1000x1000&#xff0c;长宽各放大2倍变成2000x2000。 举例&#xff1a; 原始尺寸&#xff1a; 放大2倍后 最后平均值计算成像&#xff1a; MSAA&…...

Python设计模式实战:开启软件设计的精进之旅

&#x1f31f;&#x1f31f; 欢迎来到我的技术小筑&#xff0c;一个专为技术探索者打造的交流空间。在这里&#xff0c;我们不仅分享代码的智慧&#xff0c;还探讨技术的深度与广度。无论您是资深开发者还是技术新手&#xff0c;这里都有一片属于您的天空。让我们在知识的海洋中…...

用RNN(循环神经网络)预测股票价格

RNN&#xff08;循环神经网络&#xff09;是一种特殊类型的神经网络&#xff0c;它能够处理序列数据&#xff0c;并且具有记忆先前信息的能力。这种网络结构特别适合于处理时间序列数据、文本、语音等具有时间依赖性的问题。RNN的核心特点是它可以捕捉时间序列中的长期依赖关系…...

08-图7 公路村村通(C)

很明显聪明的同学已经发现&#xff0c;这是一个稠密图&#xff0c;所以用邻接矩阵。可以很好的表达&#xff0c;比邻接表有优势&#xff0c;所以&#xff0c;采用邻接矩阵破题&#xff0c; 当然也可以用邻接表&#xff0c;仔细观察我的AC,会发现其实都一样&#xff0c;只是存储…...

Java-sleep()、wait()、join()、yield()的区别

关于线程&#xff0c;作为八股文面试中必问点&#xff0c;我们需要充分了解sleep()、wait()、join()以及yield()的区别。在正式开始之前先让我们了解两个概念&#xff1a;锁池和等待池 1.锁池 所有需要竞争同步锁的线程都会放在锁池当中&#xff0c;比如当前对象的锁已经被其中…...

Linux命令的补全和自动完成完全开启

前言 在安装好RockyLinux8.8后&#xff0c;输入dn后&#xff0c;按下“TAB”能自动提示&#xff0c;但在输入dnf make后&#xff0c;按下“TAB”不能实现自动补全&#xff0c;如果要使Linux的Bash支持完整的自动提示和补全功能&#xff0c;还需要执行一些其它操作。 内容 1、…...

Deep Active Contours for Real-time 6-DoF Object Tracking

这篇论文解决了从RGB视频进行实时6自由度&#xff08;6-DoF&#xff09;物体跟踪的问题。此前的基于优化的方法通过对齐投影模型与图像来优化物体姿态&#xff0c;这种方法依赖于手工设计的特征&#xff0c;因此容易陷入次优解。最近的基于学习的方法使用神经网络来预测姿态&am…...

IDEA安装教程配置java环境(超详细)

引言 IntelliJ IDEA 是一款功能强大的集成开发环境&#xff08;IDE&#xff09;&#xff0c;广泛用于 Java 开发&#xff0c;但也支持多种编程语言&#xff0c;如 Kotlin、Groovy 和 Scala。本文将为你提供一步一步的指南&#xff0c;帮助你在 Windows 系统上顺利安装 Intelli…...

Excel文档的读取(1)

熟悉使用Excel的同学应该都知道&#xff0c;在单个Excel表格里想要分商品计算总销售额&#xff0c;使用数据透视表也可以非常快速方便的获得结果。但当有非常大量的Excel文件需要处理时&#xff0c;每一个Excel文件单独去做数据透视也会消耗大量的时间。就算使用Power Query这样…...

Linux:体系结构和操作系统管理

目录 一、冯诺依曼体系结构 1.问题1 2.问题2 二、操作系统管理 一、冯诺依曼体系结构 本章将会谈论一下对冯诺依曼计算机体系结构的理解。 在2024年&#xff0c;几乎所有的计算机&#xff0c;都遵守冯诺依曼体系结构。 冯诺依曼体系结构是应用在硬件层面的&#xff0c;而硬…...

c++ install boost lib

同步系统上的软件包列表 sudo apt-update 整个库安装: sudo apt-get install libboost-all-dev 安装部分库: sudo apt-get install libboost-thread-dev sudo apt-get install libboost-filesystem-dev 链接时加上: -lboost_filesystem -lboost_system 例如: g -Wall -o bo…...

文件加密最简单的方法有哪些?十个电脑文件加密方法【超详细】

在当今数字化和信息化的时代&#xff0c;数据已成为企业最重要的资产之一。内部数据外泄不仅可能导致商业秘密的丧失&#xff0c;还可能对企业的声誉和财务健康造成严重影响。为了有效防止内部数据外泄&#xff0c;企业需要实施综合的防泄密解决方案。以下是十大最佳防泄密解决…...

IPv6地址的表示方法

IPv6地址总长度为128比特&#xff0c;通常分为8组&#xff0c;每组为4个十六进制数的形式&#xff0c;每组十六进制数间用冒号分隔。 例如&#xff1a;2409:8745:039a:c700:0000:0000:0162&#xff0c;这是IPv6地址的首选格式。 为了书写方便&#xff0c;IPv6还提供了压缩格式…...

Kubernetes 之 kubelet 与 CRI、CNI 的交互过程

序言 当一个新的 Pod 被提交创建之后&#xff0c;Kubelet、CRI、CNI 这三个组件之间进行了哪些交互&#xff1f; Kubelet -> CRI -> CNI 如上图所示&#xff1a; Kubelet 从 kube-api-server 处监听到有新的 pod 被调度到了自己的节点且需要创建。Kubelet 创建 sandbo…...

【python】OpenCV—Age and Gender Classification

文章目录 1、任务描述2、网络结构2.1 人脸检测2.2 性别分类2.3 年龄分类 3、代码实现4、结果展示5、参考 1、任务描述 性别分类和年龄分类预测 2、网络结构 2.1 人脸检测 输出最高的 200 个 RoI&#xff0c;每个 RoI 7 个值&#xff0c;&#xff08;xx&#xff0c;xx&#x…...

python安装换源

安装 python 使用演示的是python 3.8.5 安装完成后&#xff0c;如下操作打开命令行&#xff1a;同时按 “WindowsR” > 输入 “cmd” -> 点击确定 python换源 临时换源&#xff1a; #清华源 pip install markdown -i https://pypi.tuna.tsinghua.edu.cn/simple # 阿里…...

JavaScript练手小技巧:利用鼠标滚轮控制图片轮播

近日&#xff0c;在浏览网站的时候&#xff0c;发现了一个有意思的效果&#xff1a;一个图片轮播&#xff0c;通过上下滚动鼠标滚轮控制图片的上下切换。 于是就有了自己做一个的想法&#xff0c;顺带复习下鼠标滚轮事件。 鼠标滚轮事件&#xff0c;参考这篇文章&#xff1a;…...

搭建Eureka高可用集群 - day03

全部代码发出来了 搭建服务提供者 步骤&#xff1a; 1.创建项目&#xff0c;引入依赖 2.添加Eureka相关配置 3.添加EnableEurekaClient注解 4.测试运行 步骤1&#xff1a;创建项目&#xff0c;引入依赖 使用Spring Initializr方式创建一个名称为eureka-provider的Sprin…...

并行程序设计基础——并行I/O(2)

目录 一、显式偏移的并行文件读写 1、阻塞方式 1.1 MPI_FILE_READ_AT 1.2 MPI_FILE_WRITE_AT 1.3 MPI_FILE_READ_AT_ALL 1.4 MPI_FILE_WRITE_AT_ALL 2、非阻塞方式 2.1 MPI_FILE_IREAD_AT 2.2 MPI_FILE_IWRITE_AT 3、两步非阻塞组调用 3.1 MPI_FILE_READ_AT_ALL_BEG…...

Java三种创建多线程的方法

线程是什么&#xff1a; 进程是程序的一次动态执行的过程&#xff0c;线程是进程中执行运算最小单位&#xff0c;一个进程在其执行过程中可以产生多个线程&#xff0c;而线程必须在某个进程内执行。 如果在一个进程中同时运行了多个线程&#xff08;必须包含一个主线程&#…...

828华为云征文 | 云上私人数据管家,jMalCloud个人网盘在华为云Flexus的Docker化部署实践

华为云服务器Flexus X实例介绍 华为云Flexus云服务器X实例&#xff0c;是由国家科技进步奖获得者、华为公司Fellow、华为云首席架构师顾炯炯牵头研发。它基于擎天QingTian架构、瑶光云脑、盘古大模型等根技术创新&#xff0c;是业界首款应用驱动的柔性算力云服务器&#xff0c;…...

C# 开源教程带你轻松掌握数据结构与算法

目录 前言 项目介绍 项目特点 项目展示 1、内容导图 2、部分目录 3、源码示例 项目地址 最后 前言 在项目开发过程中&#xff0c;理解数据结构和算法如同掌握盖房子的秘诀。算法不仅能帮助我们编写高效、优质的代码&#xff0c;还能解决项目中遇到的各种难题。 给大家…...

由一个 SwiftData “诡异”运行时崩溃而引发的钩深索隐(五)

概述 在 WWDC 24 中,苹果推出了数据库框架 SwiftData 2.0 版本。其新加入的历史记录追踪(History Trace)机制着实让秃头码农们“如痴如醉”了一番。 我们在之前的博文中已经介绍了 History Trace 是如何处理数据新增操作的。而在这里,我们将再接再厉来完成数据删除时的全盘…...

python爬虫基础:了解html

编辑器vscode <!DOCTYPE html> <html><head><title>第一个网页</title></head><body><h1>字体</h1><h2>字体</h2><h3>字体</h3><p>Lorem, ipsum dolor sit amet consectetur adipisicing…...

spring security OAuth2 客户端接入gitee

一、简介 OAuth 是一个开放标准&#xff0c;该标准允许用户让第三方应用访问该用户在某一网站上存储的私密资源(如头像、照片、视频等)&#xff0c;并且在这个过程中无须将用户名和密码提供给第三方应用。通过令牌(token)可以实现这一功能&#xff0c;每一个令牌授权一个特定的…...

阿里云镜像报错 [Errno 14] HTTP Error 302 - Found 问题解决记录

1、问题背景和解决思路 在本地安装 CentOS7 后&#xff0c;网络已调通可正常上网&#xff0c;但切换阿里云镜像后&#xff0c;使用 yum 安装软件时出现 “[Errno 14] HTTPS Error 302 - Found Trying other mirror.” 报错&#xff0c;原因是 yum 源配置问题。给出了详细的解决…...

《Linux运维总结:基于X86_64+ARM64架构CPU使用docker-compose一键离线部署consul 1.18.1容器版分布式ACL集群》

总结&#xff1a;整理不易&#xff0c;如果对你有帮助&#xff0c;可否点赞关注一下&#xff1f; 更多详细内容请参考&#xff1a;《Linux运维篇&#xff1a;Linux系统运维指南》 一、部署背景 由于业务系统的特殊性&#xff0c;我们需要面向不通的客户安装我们的业务系统&…...

深入剖析嵌套调用和链式访问,以及函数的声明和定义(超全面覆盖)

1. 前情提要 在上一篇博客中&#xff0c;我们大致了解了函数的种类&#xff0c;以及自定义函数中形参和实参的具体区别 我们知道实参是需要传递给形参的&#xff0c;但其实形参和实参占据的是完全独立的内存空间 x&#xff0c;y在执行过程中会得到a和b的值&#xff0c;但是x…...