【STM32嵌入式系统设计与开发拓展】——12_Timer(定时器中断实验)

目录

  • 1、什么是定时器?
    • 定时器用于测量时间间隔,而计数器用于计数外部事件的次数
  • 2、定时器的主要功能和用途?
  • 3、定时器类型?
  • 4、定时器的编写过程
  • 5、代码分析
    • 定时器计算?
    • 计算过程
    • 周期(arr):

1、什么是定时器?

在这里插入图片描述

定时器就是计数器!!!
定时器就是计数器!!!
定时器就是计数器!!!

在 STM32 微控制器中,定时器是一种用于测量时间间隔、生成精确时间事件或执行周期性任务的硬件外设。定时器通常包含多个独立的计数器,每个计数器可以配置为不同的模式和功能,以满足各种应用需求。


定时器和计数器在许多方面是相似的,但它们有不同的用途和特点:
定时器(Timer):

主要用途:用于测量时间间隔。可以用来生成精确定时事件,例如控制LED闪烁的频率或执行周期性任务。
工作原理:通常以固定频率递增或递减计数值。当达到预设值时,定时器会产生一个中断或信号,通知处理器或其他部件时间已经到达。
例子:在微控制器中,你可以使用定时器来每秒钟中断一次,执行特定任务。

计数器(Counter):
主要用途:用于计数外部事件或信号的次数,例如检测一个输入引脚上信号的上升沿或下降沿的数量。
工作原理:每次检测到预定义的事件(如上升沿、下降沿或脉冲)时,计数器的值会递增或递减。计数器也可以产生中断,当达到预设值时,通知处理器进行处理。
例子:在微控制器中,你可以使用计数器来计算外部按钮被按下的次数。

定时器用于测量时间间隔,而计数器用于计数外部事件的次数


2、定时器的主要功能和用途?

1、计时:定时器可以用来测量时间间隔或延时。通过配置定时器的预分频器和自动重装载寄存器,可以实现精确的时间测量。

2、脉宽调制(PWM):定时器可以生成 PWM 信号,用于控制电机、LED 亮度或其他需要调节占空比的应用。

3、事件计数:定时器可以用于计数外部事件的次数,比如按钮按下的次数或传感器的脉冲。

4、输入捕获:定时器可以捕获输入信号的边沿,用于测量信号的频率或周期。

6、输出比较:定时器可以生成定时事件,比如在特定时间输出一个信号或触发中断。

3、定时器类型?

在这里插入图片描述

4、定时器的编写过程

1、初始化定时器:设置预分频器和自动重装载寄存器,配置定时器的基本参数。
2、配置定时器模式:根据需要选择计时模式、PWM 模式、输入捕获模式或输出比较模式。
3、启用定时器中断(如需要):配置并启用定时器的中断功能。
4、启动定时器:使能定时器,使其开始计数或生成 PWM 信号。

5、代码分析

定时器计算?

在这里插入图片描述

1M = 1000K
1K = 1000

1s = 1000ms
1ms = 1000us

	SystemTinerInit(1000-1,3600-1); //系统时间初始化 定时100ms
公式:
Tout=((arr+1)*(psc+1))/Ft us,Ft=定时器工作频率,单位:Mhz

Tout 是定时器溢出时间。
arr 是自动重装寄存器的值(即1000-1)。
psc 是预分频器的值(即3600-1)。
𝐹𝑡 是定时器的工作频率,单位为兆赫兹(MHz)。

1000*3600/36M = 3 600 000/36 000 000 = 0.1s = 100ms


等待计时

开始|+--> 等待时间(gTimer)为0吗?|        ||        +-- 是 --> 返回1 (超时)|        ||        +-- 否|+--> 计算定时器剩余时间 (GTr = SystemTimer % gTimer)|+--> 定时器剩余时间为0且未检测到超时且当前定时器时间不等于上次记录的时间吗?|        ||        +-- 是 --> 设置Rti为1,更新Gti为gTimer,返回1 (超时)|        ||        +-- 否|+--> 定时器剩余时间不为0且上次检测到超时吗?|        ||        +-- 是 --> 设置Rti为0|        ||        +-- 否|+--> 记录时间为0吗?|        ||        +-- 是 --> 将GetTimer设置为当前系统时间 (SystemTimer)|        ||        +-- 否|+--> 系统时间减去开始时间等于等待时间吗?|        ||        +-- 是 --> 将GetTimer设置为0,返回1 (超时)|        ||        +-- 否 --> 返回0 (未超时)
uint8_t WaitTimerOut(uint32_t gTimer)
{	uint32_t GTr = 0;                         // 定义变量用于存储定时器剩余时间if(gTimer==0) return 1;                   // 如果等待时间为0,则直接返回1,表示不等待GTr = SystemTimer%gTimer;	                // 计算定时器剩余时间	if((GTr==0)&&(!Rti)&&(Gti!=gTimer))       // 如果定时器剩余时间为0,且上次未检测到超时,并且当前定时器时间不等于上次记录的时间 { Rti=1; Gti = gTimer; return 1;}         // 设置标志表示检测到超时,更新时间返回1表示超时else if((GTr!=0)&&(Rti))                  // 如果时间不为0,且上次检测到超时,则将标志置为0Rti=0;if(!GetTimer) GetTimer = SystemTimer;     // 如果记录时间为0,则将其设置为当前系统时间if(SystemTimer-GetTimer==gTimer)          // 如果系统时间减去开始时间等于等待时间,则返回1表示超时{ GetTimer = 0; return 1; }return 0;
}

开始|+--> 中断状态为溢出中断吗? (TIM_GetITStatus(TIM3, TIM_IT_Update) == SET)|        ||        +-- 否 --> 跳过处理|        ||        +-- 是|+--> 系统时间计数器加1 (SystemTimer++)|+--> 系统时间计数器达到60吗? (SystemTimer == 60)|        ||        +-- 否 --> 继续处理|        ||        +-- 是|              ||              +--> 将系统时间计数器重置为0 (SystemTimer = 0)|              ||              +--> 将记录的定时器开始时间清零 (GetTimer = 0)|+--> 清除定时器更新中断标志位 (TIM_ClearITPendingBit(TIM3, TIM_IT_Update))|
结束
void TIM3_IRQHandler(void)   
{	if(TIM_GetITStatus(TIM3,TIM_IT_Update)==SET) //溢出中断{SystemTimer++;                             // 系统时间计数器加1if(SystemTimer==60)	                       // 如果系统时间计数器达到60,则重置为0,并且清零记录的定时器开始时间	{	SystemTimer=0;GetTimer = 0;}}/* 清除定时器更新中断标志位 */	TIM_ClearITPendingBit(TIM3,TIM_IT_Update);  //清除中断标志位
}

让我们重新计算一下。如果定时器的时钟频率是 36 MHz,预分频器(psc)为 3599,周期(arr)为 999,那么定时器中断的时间间隔如下:

计算过程

预分频器(psc):
预分频器将时钟频率从 36 MHz 分频到
36MHz/3600=10kHz

周期(arr):

计数器从 0 计数到 999,即 1000 个计数。
计数频率为 10 kHz,所以计数 1000 次所需的时间为
1000/10kHz=0.1秒即 100 毫秒。

因此,每次定时器中断的时间间隔是 100 毫秒。所以 SystemTimer 在每次中断服务程序 (TIM3_IRQHandler) 被调用时增加一次,也就是每 100 毫秒增加一次。
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/1487441.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

Apollo使用(3):分布式docker部署

Apollo 1.7.0版本开始会默认上传Docker镜像到Docker Hub,可以按照如下步骤获取 一、获取镜像 1、Apollo Config Service 获取镜像 docker pull apolloconfig/apollo-configservice:${version} 我事先下载过该镜像,所以跳过该步骤。 2、Apollo Admin S…

Leetcode3219. 切蛋糕的最小总开销 II

Every day a Leetcode 题目来源:3219. 切蛋糕的最小总开销 II 解法1:贪心 谁的开销更大,就先切谁,并且这个先后顺序与切的次数无关。 代码: /** lc appleetcode.cn id3219 langcpp** [3219] 切蛋糕的最小总开销 I…

医疗信息化之PACS系统源码,C#医学影像系统源码,成熟在用稳定运中

C#语言开发的一套PACS系统源码,C/S架构,成熟稳定,多家大型综合医院应用案例。自主版权,支持二次开发,授权后可商用。 医学影像存储与传输系统是针对数据库存储、传输服务、图像处理进行了优化,存储更安全、传输更稳定、…

人工智能与机器学习原理精解【6】

文章目录 数值优化基础理论凹凸性定义在国外与国内存在不同国内定义国外定义总结示例与说明注意事项 国内凹凸性二阶定义的例子凹函数例子凸函数例子 凸函数(convex function)的开口方向凸函数的二阶导数凸函数的二阶定义单变量函数的二阶定义多变量函数…

C# 知识点总结

入门 C#程序在.NET上运行,.NET framework包含两个部分: ①:.NET framework类库 ②:公共语言运行库CLR(.NET虚拟机) CLS(公共语言规范) CTS(通用类型系统) .N…

【Node.js基础04】包的理解与使用

一:包的理解与简介 1 什么是包 包是一个将模块、代码、以及其他资料聚合成的文件夹 2 包的分类 项目包:编写项目代码的文件夹 软件包:封装工具和方法供开发者使用 3 为什么要在软件包中编写package.json文件 记录包的清单信息 二&…

09-optee内核-线程处理

快速链接: . 👉👉👉 个人博客笔记导读目录(全部) 👈👈👈 付费专栏-付费课程 【购买须知】:【精选】TEE从入门到精通-[目录] 👈👈👈线程处理 OP-TEE内核使用几个线程来支持在 并行(未完全启用!有用于不同目的的处理程序。在thread.c中,您将找到一个名为…

Binius-based zkVM:为Polygon AggLayer开发、FPGA加速的zkVM

1. 引言 近日,ZK硬件加速巨头Irreducible和Polygon团队宣布联合开发生产级的Binius-based zkVM,用于助力Polygon AggLayer,实现具有低开销、硬件加速的binary proofs。 Irreducible(曾用名为Ulvetanna)团队 Benjamin …

JavaSE从零开始到精通(八) - 单列集合类

1. 集合概述 集合类是一种用于存储对象的容器,Java 提供了一组预定义的集合类,它们位于 java.util 包中。这些集合类提供了各种数据结构,包括列表、集合、队列、映射等,每种数据结构都有其独特的特性和用途。 Java 中的集合类是…

secureCRT同时在所有已打开窗口执行命令、mac-os下使用的SecureCRT版本 以及 SecureCRT一段时间不操作没有响应的问题

一、secureCRT命令行工具一次性同时在所有已打开窗口执行命令 公司的服务器比较多,最近因为opcache,上线发布后,需要重启所有的WEB服务器上的php。目前使用的jenkins发布,不过账号安全问题,给jenkins的账号权限受限不能…

【C++中线程学习】

1、多线程 C11之前没有引入线程的概念&#xff0c;如果想要实现多线程&#xff0c;需要借助操作系统平台提供的API&#xff0c;比如Linux的<pthead.h>&#xff0c;或者windows下的<windows.h>。 C11提供了语言层面上的多线程&#xff0c;包含在头文件<thread.h…

雪花算法 集群uid重复问题 uid-generator-spring-boot-starter

1、在生成环境 在某个业务使用该插件生成uid,由于业务整合了 mybatis-plus模块 2、该业务是分部署集群部署以及使用的多线程获取uid&#xff0c;使用中发现唯一建冲突&#xff0c;生成的uid有重复。 然后查看日志发现 workerId 始终为0 怀疑是生成workerId出了问题。 查看跟…

Cadence23学习笔记(十四)

ARC就是圆弧走线的意思&#xff1a; 仅打开网络的话可以只针对net进行修改走线的属性&#xff1a; 然后现在鼠标左键点那个走线&#xff0c;那个走线就会变为弧形&#xff1a; 添加差分对&#xff1a; 之后&#xff0c;分别点击两条线即可分配差分对&#xff1a; 选完差分对之后…

使用Spring Boot与Spire.Doc实现Word文档的多样化操作

​ 博客主页: 南来_北往 系列专栏&#xff1a;Spring Boot实战 前言 使用Spring Boot与Spire.Doc实现Word文档的多样化操作具有以下优势&#xff1a; 强大的功能组合&#xff1a;Spring Boot提供了快速构建独立和生产级的Spring应用程序的能力&#xff0c;而Spire.Doc则…

docker产生日志过大优化

1、Docker容器启动后日志存放位置 #cat /var/lib/docker/containers/容器ID/容器ID-json.log #echo >/var/lib/docker/containers/容器ID/容器ID-json.log临时清除日志 注&#xff1a;echo一个空进去&#xff0c;不需要重启容器&#xff0c;但如果你直接删除这个日志&…

如何在vscode中对在服务器上多卡运行的bash脚本进行debug?

问题描述 使用vscode可以很方便地添加断点&#xff0c;进行代码调试。 在使用服务器时&#xff0c;我们的python代码通常是通过bash脚本来执行的&#xff0c;那么如何进行debug呢&#xff1f; 待运行的bash 脚本示例 前半段定义了一些参数&#xff0c;后半段是执行python代码…

MiniExcel:.NET中处理Excel的高效方案

在.NET开发环境中&#xff0c;处理Excel文件是一项常见的任务&#xff0c;无论是数据导入、导出还是报表生成。传统的解决方案可能存在性能瓶颈或功能限制。MiniExcel作为一个现代、高效的库&#xff0c;为.NET开发者提供了一个强大的工具来简化Excel操作。本文将介绍MiniExcel…

萝卜快跑:自动驾驶的先锋与挑战

萝卜快跑&#xff1a;自动驾驶的先锋与挑战 近段时间&#xff0c;由萝卜快跑引发的自动驾驶事件如火如荼&#xff0c;成为科技领域的热门话题。萝卜快跑作为自动驾驶领域的重要参与者&#xff0c;其最新事件引发了广泛的关注和讨论。 萝卜快跑是百度推出的自动驾驶出行服务平台…

【Socket 编程】应用层自定义协议与序列化

文章目录 再谈协议序列化和反序列化理解 read、write、recv、send 和 tcp 为什么支持全双工自定义协议网络计算器序列化和反序列化 再谈协议 协议就是约定&#xff0c;协议的内容就是约定好的某种结构化数据。比如&#xff0c;我们要实现一个网络版的计算器&#xff0c;客户端…

LeetCode算法——滑动窗口矩阵篇

1、长度最小的子数组 题目描述&#xff1a; 解法&#xff1a; 设一个 for 循环来改变指向窗口末尾的指针&#xff0c;再不断抛弃当前窗口内的首元素 最终确定满足条件的最小长度 class Solution { public:int minSubArrayLen(int target, vector<int>& nums) {int …