ESP32-C3实现定时器的启停(Arduino IDE)

devtools/2024/10/20 18:28:59/

1概述

ESP32-C3微控制器有多个定时器,这些定时器可用于各种用途,包括计时、生成PWM信号、测量输入信号的频率等。以下是ESP32-C3上可用的定时器资源:

  1. 两个硬件定时器

    • 定时器0:这是一个通用定时器,通常用于操作系统的任务调度。
    • 定时器1:同样是一个通用定时器,也可以用于其他目的。
  2. 八个LED控制定时器

    • 这些定时器专门用于LED控制,例如,它们可以用来生成PWM信号来控制LED的亮度。
  3. 两个I2S DMA定时器

    • 这些定时器用于I2S(Inter-IC Sound)接口的DMA(Direct Memory Access)操作。
  4. 一个RMT(Remote Control)定时器

    • RMT定时器用于生成和接收红外遥控信号。

这些定时器资源在ESP32-C3上的分配和功能可能会根据具体的应用和ESP-IDF(Espressif IoT Development Framework)的版本有所不同。硬件定时器(定时器0和定时器1)通常用于最关键的计时任务,而其他定时器则可用于特定外设的控制。

2实现定时器的启停(1秒)

#include <Arduino.h>hw_timer_t *timer = NULL;
volatile int seconds = 0;
volatile bool printFlag = false;  // 添加一个标志来指示是否需要打印// 定时器中断服务例程
void IRAM_ATTR tim1Interrupt() {seconds++;  // 每次中断增加秒数printFlag = true;  // 设置打印标志
}void setup() {Serial.begin(115200);timer = timerBegin(0, 80, true);  // 初始化定时器timerAttachInterrupt(timer, &tim1Interrupt, true);  // 绑定中断函数timerAlarmWrite(timer, 1000000, true);  // 设置报警值为 1 秒timerAlarmEnable(timer);  // 启用定时器中断
}void loop() {if (printFlag) {Serial.print("Seconds passed: ");Serial.println(seconds);printFlag = false;  // 重置打印标志}
}

3注意串口不能放在定时器中断内部

,放进去会产生内存溢出

  1. 串行缓冲区溢出:如果在 Serial.print 或 Serial.println 调用之间有太多的数据要发送,而串行通信速度跟不上,串行缓冲区可能会溢出,导致数据丢失或混乱。

  2. 中断处理和串行通信冲突:由于 Serial.print 和 Serial.println 不是原子操作,如果在它们执行过程中发生中断,可能会导致输出不完整或混乱。

  3. 其他代码或库的干扰:如果在您的程序中还有其他代码或库使用串行端口进行通信,它们可能会与您的打印语句冲突。

4重点解释

  1. timerBegin(0, 80, true);

    • timerBegin 函数用于初始化定时器,并返回一个定时器句柄。
    • 第一个参数 0 指定要使用的定时器组。ESP32-C3 有两个定时器组,每个组有两个定时器。这里选择定时器组0。
    • 第二个参数 80 是预分频器的值。定时器的时钟源通常是 APB(Advanced Peripheral Bus)时钟,预分频器用于降低时钟频率。预分频器值设置为80,意味着定时器的时钟频率是 APB 时钟的 1/80。
    • 第三个参数 true 表示定时器是向上计数的,即从0开始,每次计数增加,直到达到设定的报警值。
  2. timerAttachInterrupt(timer, &tim1Interrupt, true);

    • timerAttachInterrupt 函数用于将中断服务例程(ISR)与定时器中断关联起来。
    • 第一个参数 timer 是由 timerBegin 返回的定时器句柄。
    • 第二个参数 &tim1Interrupt 是指向中断服务例程的指针。当定时器达到报警值时,将调用此函数。
    • 第三个参数 true 表示在中断服务例程执行期间,定时器中断应该被禁用。这是为了避免在中断服务例程执行期间再次进入中断。
  3. timerAlarmWrite(timer, 1000000, true);

    • timerAlarmWrite 函数用于设置定时器的报警值,即定时器达到该值时触发中断。
    • 第一个参数 timer 是定时器句柄。
    • 第二个参数 1000000 是定时器的报警值,单位是定时器时钟周期的个数。由于定时器时钟频率是 APB 时钟的 1/80,因此 1000000 个周期大约等于 1 秒(假设 APB 时钟频率为 80MHz)。
    • 第三个参数 true 表示报警值设置后,定时器会自动重载并重新开始计数,从而实现周期性中断。
  4. timerAlarmEnable(timer);

    • timerAlarmEnable 函数用于启用定时器中断。
    • 参数 timer 是定时器句柄。
    • 一旦启用,定时器开始计数,当达到设定的报警值时,将调用之前关联的中断服务例程。

时间计算

以下是计算步骤:

  1. 确定 APB 时钟频率: ESP32-C3 的 APB 时钟频率默认为 80 MHz(即每秒 80,000,000 周)。

  2. 应用预分频器: 您设置的预分频器值为 80,这意味着定时器时钟频率是 APB 时钟频率的 1/80。

    定时器时钟频率 = APB 时钟频率 / 预分频器值
    定时器时钟频率 = 80 MHz / 80 = 1 MHz (即每秒 1,000,000 周)
    
  3. 计算定时器计数周期: 您设置的报警值为 1,000,000,这是定时器达到后触发中断的计数值。

    计时时间 = 报警值 / 定时器时钟频率
    计时时间 = 1,000,000 / 1,000,000 = 1 秒

 

1分钟

#include <Arduino.h>hw_timer_t *timer = NULL;
volatile int minutes = 0;
volatile bool printFlag = false;  // 添加一个标志来指示是否需要打印// 定时器中断服务例程
void IRAM_ATTR tim1Interrupt() {minutes++;  // 每次中断增加分钟数printFlag = true;  // 设置打印标志
}void setup() {Serial.begin(115200);timer = timerBegin(0, 80, true);  // 初始化定时器timerAttachInterrupt(timer, &tim1Interrupt, true);  // 绑定中断函数timerAlarmWrite(timer, 60000000, true);  // 设置报警值为 60 秒(即 1 分钟)timerAlarmEnable(timer);  // 启用定时器中断
}void loop() {if (printFlag) {Serial.print("Minutes passed: ");Serial.println(minutes);printFlag = false;  // 重置打印标志}
}

30分钟 

#include <Arduino.h>hw_timer_t *timer = NULL;
volatile int minutes = 0;
volatile bool printFlag = false;  // 添加一个标志来指示是否需要打印// 定时器中断服务例程
void IRAM_ATTR tim1Interrupt() {minutes += 30;  // 每次中断增加30分钟数printFlag = true;  // 设置打印标志
}void setup() {Serial.begin(115200);timer = timerBegin(0, 80, true);  // 初始化定时器timerAttachInterrupt(timer, &tim1Interrupt, true);  // 绑定中断函数timerAlarmWrite(timer, 1800000000, true);  // 设置报警值为 1800 秒(即 30 分钟)timerAlarmEnable(timer);  // 启用定时器中断
}void loop() {if (printFlag) {Serial.print("Minutes passed: ");Serial.println(minutes);printFlag = false;  // 重置打印标志}
}


http://www.ppmy.cn/devtools/127350.html

相关文章

纯血鸿蒙!

纯血鸿蒙&#xff0c;这是哪个营销大师给起的名字啊&#xff01; 纯血&#xff01;象征着高贵、自信、自主、血性、英雄气概&#xff0c;都融入这纯血鸿蒙了&#xff01; 鸿蒙本就是开天辟地&#xff0c;加上纯血&#xff0c;真是荡气回肠&#xff01; 鸿蒙的推出背景 我们前…

P2-3与P2-4.【C语言基本数据类型、运算符和表达式】第三节与第四节

讲解视频&#xff1a; P2-3.【基本数据类型、运算符和表达式】第三节 P2-4.【基本数据类型、运算符和表达式】第四节 目录 必备知识与理论 任务实施 必备知识与理论 C语言中把除了控制语句和输入输出以外的几乎所有的基本操作都作为运算符处理。 其运算符和表达式数量之多&a…

Gin框架操作指南12:完结篇

Gin框架的功能确实非常丰富&#xff0c;使用postman软件确实很方便&#xff0c;省去了自己写前端代码的过程。本文回顾2-11章的内容以及使用postman软件需要注意的细节。 指南2&#xff1a;JSON渲染。演示AsciiJSON JSONP PureJSON SecureJSON XML-JSON-YAML-ProtoBuf渲染。 …

《YOLO1》论文精读:第一次实现端到端的目标检测

You Only Look Once: Unified, Real-Time Object Detection论文地址Paperwithcode在目标检测的排名Github参考代码&#xff0c;100k个star 该论文最新由Facebook AI发表于2016年5月&#xff0c;截止现在2024年10月&#xff0c;引用数是52854次 文章目录 论文核心内容&#xff…

【Hive】4-HiveSQL 数据操控语言(DML)

HiveSQL 数据操控语言&#xff08;DML&#xff09; load加载数据 语法 -- 语法规则 LOAD DATA [LOCAL] INPATH filepath [OVERWRITE] INTO TABLE tablename [PARTITION (partcol1val1, partcol2val2...)] -- 语法规则&#xff08;Hive 3.0及以后&#xff09; LOAD DATA [LOC…

Vue的响应式原理

Vue.js 是一个流行的前端框架&#xff0c;它的响应式系统是其核心特性之一&#xff0c;能够有效地处理数据变化并自动更新视图。在这篇文章中&#xff0c;我们将探讨 Vue 的响应式原理&#xff0c;包括其实现方式、关键概念以及性能优化。 1. 响应式原理概述 Vue 的响应式原理…

ubuntu 安装haproxy

####安装##### sudo apt update sudo apt install haproxy sudo haproxy -v sudo systemctl status haproxy sudo cp /etc/haproxy/haproxy.cfg /etc/haproxy/haproxy.cfg-org####配置站点##### nano /etc/haproxy/haproxy.cfgfrontend www-httpbind *:5001mode httpdefault_ba…

即使是编程新手,也能利用ChatGPT编写高质量的EA

在外汇交易领域&#xff0c;MetaTrader是一款备受欢迎的交易软件&#xff0c;包括MT5和MT4&#xff0c;提供了众多强大的分析工具和自动化交易功能。对于没有编程经验的新手而言&#xff0c;编写专家顾问&#xff08;EA&#xff09;可能显得既复杂又令人望而却步。幸运的是&…