ESP32 网络计时器,包含自动保存
简介
本代码是基于ESP32开发板实现的一个计时器功能,具备倒计时、计时器时长选择、显示当前时间、有源蜂鸣器报警等功能。代码中使用了WiFi网络连接、NTP时间同步、EEPROM存储等功能。通过按钮控制计时器的开始、停止和计时器时长的选择。
运行原理概述
在ESP32开发板上,使用了三个按钮,分别为开始计时按钮(BUTTON1)、停止计时按钮(BUTTON2)和计时器时长选择按钮。首先连接WiFi网络,同步NTP时间,并初始化OLED屏幕。
然后通过setdefaulttime()函数设置计时器默认时长,并初始化按钮引脚。
在主循环中通过checkTimerButtons()函数检测按钮状态,若按钮被按下,根据按钮类型执行相应的操作。如果开始计时按钮被按下,则开始计时,并在OLED屏幕上显示计时器倒计时时间。
如果停止计时按钮被按下,则停止计时,并根据按钮状态执行相应的操作。如果计时器时长选择按钮被按下,则切换计时器时长,并在OLED屏幕上显示新的计时器时长。
特色功能
1.具备倒计时功能,可以按照设定的计时器时长进行倒计时,到时后有报警提示。
2.计时器时长可以选择,用户可以根据需求选择不同的计时器时长。
3.通过WiFi网络连接和NTP时间同步功能,保证了计时器的时间准确性。
4.使用EEPROM存储功能,保存计时器时长选择的状态,即使重新上电,也能保留上一次的时长选择状态。
重要函数的解释
-
void setdefaulttime()函数:设置计时器默认时长,并从EEPROM中读取上一次的时长选择状态。
-
void checkTimerButtons()函数:检测按钮状态,并根据按钮类型执行相应的操作。
-
void updateTimeDisplay()函数:更新OLED屏幕上的时间和计时器倒计时时间。
-
bool saveIntToEEPROM(int value, int address)函数:将整数值存储到EEPROM中。
-
int readIntFromEEPROM(int address)函数:从EEPROM中读取整数值。
| ESP32开发板引脚 | 硬件设备引脚 | 连接方式 |
|---|---|---|
| GPIO21 | OLED_SDA | I2C数据线 |
| GPIO22 | OLED_SCL | I2C时钟线 |
| GPIO35 | BUTTON1 | 按钮输入 |
| GPIO34 | BUTTON2 | 按钮输入 |
| GPIO32 | BUZZER_PIN | 有源蜂鸣器引脚 |
| 3.3V | OLED_VCC | OLED显示屏电源 |
| GND | OLED_GND | OLED显示屏电源和地 |
| 3.3V | BUTTON1 | 按钮电源 |
| 3.3V | BUTTON2 | 按钮电源 |
| GND | BUTTON1 | 按钮电源和地 |
| GND | BUTTON2 | 按钮电源和地 |
#include <WiFi.h>
#include <NTPClient.h>
#include <Wire.h>
#include <Adafruit_SSD1306.h>
#include "soc/soc.h"
#include "soc/rtc_cntl_reg.h"
#include <EEPROM.h>
#define OLED_ADDR 0x3C // OLED屏幕地址
#define OLED_SDA 21 // ESP32开发板上的SDA引脚
#define OLED_SCL 22 // ESP32开发板上的SCL引脚
#define BUTTON1 35 // 开始计时按钮
#define BUTTON2 34 // 停止计时按钮
#define BUZZER_PIN 32 // 有源蜂鸣器连接的引脚
#define MAXMENU 5
#define ADDR_1 1
WiFiUDP ntpUDP;
NTPClient timeClient(ntpUDP, "ntp.ntsc.ac.cn"); // NTP客户端
Adafruit_SSD1306 display(128, 64, &Wire, -1); // OLED屏幕
int menuvalue = 1;
// 计时器相关变量
unsigned long startTime = 0; // 计时器开始时间
unsigned long stopTime = 0; // 计时器停止时间
bool timerRunning = false; // 是否正在计时
unsigned long duration = 0; // 计时器持续时间
unsigned long remainingTime = 0; // 剩余时间// 时间显示相关变量
String lastFormattedTime = ""; // 上次更新的时间
unsigned long lastUpdateTime = 0; // 上次更新时间的时间戳
// 默认倒计时时间
int defaulttime = 5*60*1000;
// 设置默认倒计时时间
void setdefaulttime();bool saveIntToEEPROM(int value, int address) {EEPROM.begin(16);EEPROM.put(address, value);bool success = EEPROM.commit();EEPROM.end();display.setCursor(0, 50);display.setTextSize(1);if(success){display.println("Save Success");}else {display.println("Failed");}Serial.print("save menuvalue:");Serial.println(value);return success;
}int readIntFromEEPROM(int address) {if (EEPROM.begin(16)) {int value;if (EEPROM.get(address, value)) {EEPROM.end();return value;} else {Serial.println("Failed to read from EEPROM");}} else {Serial.println("EEPROM initialization failed");}EEPROM.end();return 0;
}unsigned long timerDuration = 5 * 60 * 1000; // 初始计时器长度为5分钟//frequency参数是需要发出的声音频率,单位为Hz;duration参数是发出声音的持续时间,单位为毫秒。
void buzz(int frequency, long duration)
{int period = 1000000 / frequency; // 计算周期int pulse = period / 2; // 计算脉冲时间for (long i = 0; i < duration * 1000L; i += period){digitalWrite(BUZZER_PIN, HIGH); // 发送高电平delayMicroseconds(pulse); // 持续脉冲时间的一半digitalWrite(BUZZER_PIN, LOW); // 发送低电平delayMicroseconds(pulse); // 持续脉冲时间的一半}
}
void checkTimerButtons()
{if (digitalRead(BUTTON1) == LOW && !timerRunning){// 开始计时器buzz(2000, 100);startTime = millis();timerRunning = true;}if (digitalRead(BUTTON2) == LOW){buzz(2000, 100);if (timerRunning){// 停止计时器stopTime = millis();timerRunning = false;remainingTime = timerDuration - (stopTime - startTime);while(digitalRead(BUTTON2) == LOW);delay(100);}else{// 切换计时器时间长度if (menuvalue == 1){timerDuration = 10 * 60 * 1000;menuvalue =2;}else if (menuvalue == 2){timerDuration = 30 * 60 * 1000;menuvalue=3;}else if (menuvalue==3){timerDuration = 60 * 60 * 1000;menuvalue=4;}else if (menuvalue ==4){timerDuration = 5 * 60 * 1000;menuvalue = 1;}remainingTime = timerDuration;defaulttime = timerDuration;saveIntToEEPROM(menuvalue,ADDR_1);while(digitalRead(BUTTON2) == LOW);}}
}void setdefaulttime()
{menuvalue = readIntFromEEPROM(ADDR_1);Serial.print("read menuvalue:");Serial.println(menuvalue);if (menuvalue>0&&menuvalue<=MAXMENU){switch(menuvalue){case 1:defaulttime = 5 * 60 * 1000;break;case 2:defaulttime = 10 * 60 * 1000;break;case 3:defaulttime = 30 * 60 * 1000;break;case 4:defaulttime = 60 * 60 * 1000;break;default:defaulttime = 5 * 60 * 1000;break;}}else{menuvalue = 1;saveIntToEEPROM(menuvalue,ADDR_1);defaulttime = 5 * 60 * 1000;}remainingTime = defaulttime;}
void updateTimeDisplay()
{timeClient.update();String formattedTime = timeClient.getFormattedTime();display.clearDisplay();display.setCursor(0, 0);display.setTextSize(2);display.setTextColor(WHITE);display.println(formattedTime);if (timerRunning){duration = millis() - startTime;remainingTime = timerDuration - duration;display.setCursor(0, 36);display.setTextSize(4);int minutes = remainingTime / 1000 / 60;int seconds = (remainingTime / 1000) % 60;if (minutes < 10){display.print("0");}display.print(minutes);display.print(":");if (seconds < 10){display.print("0");}display.println(seconds);}else{display.setCursor(0, 36);display.setTextSize(4);int _minutes = remainingTime / 1000 / 60;int _seconds = (remainingTime / 1000) % 60;if (_minutes < 10){display.print("0");}display.print(_minutes);display.print(":");if (_seconds < 10){display.print("0");}display.println(_seconds);}display.display();
}void setup()
{// 连接WiFi网络pinMode(BUZZER_PIN, OUTPUT);WRITE_PERI_REG(RTC_CNTL_BROWN_OUT_REG, 0); Serial.begin(115200);Wire.begin(OLED_SDA, OLED_SCL);display.begin(SSD1306_SWITCHCAPVCC, OLED_ADDR);display.clearDisplay();display.setCursor(0, 0);display.setTextSize(1);display.setTextColor(WHITE);display.println("Connecting to ");display.println("360WiFi-016C34");display.println("Password: ");display.println("wangjinxuan");display.display();WiFi.begin("360WiFi-016C34", "wangjinxuan");int i = 0;while (WiFi.status() != WL_CONNECTED && i < 10){delay(500);display.clearDisplay();display.setCursor(0, 0);display.print("Connecting");for (int j = 0; j < i; j++){display.print(".");}display.display();i++;}if (WiFi.status() != WL_CONNECTED){display.clearDisplay();display.setCursor(0, 0);display.println("Failed to connect.");display.display();while (true){// 连接失败,停止程序}}// 初始化OLED屏幕Wire.begin(OLED_SDA, OLED_SCL);display.begin(SSD1306_SWITCHCAPVCC, OLED_ADDR);display.clearDisplay();//初始化数值setdefaulttime();// 初始化按钮引脚pinMode(BUTTON1, INPUT_PULLUP);pinMode(BUTTON2, INPUT_PULLUP);// 同步时间timeClient.begin();timeClient.setTimeOffset(28800); // 设置时区(这里为东八区)
}void loop()
{checkTimerButtons();updateTimeDisplay();if (timerRunning){duration = millis() - startTime;remainingTime = timerDuration - duration;if (remainingTime <= 0 || remainingTime>timerDuration){// 计时器已完成timerRunning = false;buzz(2000,1000);delay(800);buzz(2000,1000);delay(800);buzz(2000,1000);delay(800);remainingTime = timerDuration;// TODO: 执行计时器完成操作}}
}
相关文章:
ESP32 网络计时器,包含自动保存
简介 本代码是基于ESP32开发板实现的一个计时器功能,具备倒计时、计时器时长选择、显示当前时间、有源蜂鸣器报警等功能。代码中使用了WiFi网络连接、NTP时间同步、EEPROM存储等功能。通过按钮控制计时器的开始、停止和计时器时长的选择。 运行原理概述 在ESP32开…...
【ChatGPT】阿里版 ChatGPT 突然官宣意味着什么?
Yan-英杰的主页 悟已往之不谏 知来者之可追 C程序员,2024届电子信息研究生 目录 阿里版 ChatGPT 突然官宣 ChatGPT 技术在 AI 领域的重要性 自然语言生成 上下文连续性 多语言支持 ChatGPT 未来可能的应用场景 社交领域 商业领域 编辑 医疗领域…...
IPEmotion控制模块-PID循环应用
IPEmotion专业版、开发版支持控制模块,并且该模块支持函数发生器、PID控制器、路由器、序列控制和序列控制块以及参考曲线生成器。本文主要针对PID(P:Proportional control 比例控制;I:Integral control 积分控制&…...
【元分析研究方法】学习笔记2.检索文献(含100种学术文献搜索清单链接)
检索文献 该步骤的作用该步骤中需要注意的问题该步骤中部分知识点我的收获 参考来源:库珀 (Cooper, H. M. )., 李超平, & 张昱城. (2020). 元分析研究方法: A step-by step approach. 中国人民大学出版社. 该步骤的作用 1.识别相关文献的来源; 2.识别…...
题目:16版.自由落体
1、实验要求 本实验要求:模拟物体从10000米高空掉落后的反弹行为。 1-1. 创建工程并配置环境: 1-1.1. 限制1. 工程取名:SE_JAVA_EXP_E009。 1-1.2. 限制2. 创建包,取名:cn.campsg.java.experiment。 1-1.3. 限制3. 创建…...
视频可视化搭建项目,通过简单拖拽方式快速生产一个短视频
一、开源项目简介 《视搭》是一个视频可视化搭建项目。您可以通过简单的拖拽方式快速生产一个短视频,使用方式就像易企秀或百度 H5 等 h5 搭建工具一样的简单。目前行业内罕有关于视频可视化搭建的开源项目,《视搭》是一个相对比较完整的开源项目&#…...
network-1 4 layer internet model
4layer model applicationtransport tcp: transmission control protocol enable correct in-order delivery of data, running on top of the network layer service.udp: user datagram protocolnetwork packet:data、from、tonetwork->linkiplink source en…...
计算机网络笔记(横向)
该笔记也是我考研期间做的整理。一般网上的笔记是按照章节纪录的,我是按照知识点分类纪录的,大纲如下: 文章目录 1. 各报文1.1 各报文头部详解1.2 相关口诀 2. 各协议2.1 各应用层协议使用的传输层协议与端口2.2 各协议的过程2.2.1 数据链路层…...
0.redis-实践
1.redis内存设置多少,默认是0,不限制 2.如何配置,修改内存大小 1) 查看最大占用内存 # maxmeory <bytes> 或者 config get maxmemory 2) 默认内存多少可以用: 64位系统下不限制,32位下最多3G 3) 如何配置: 默认总内存的3/4 4) 如何修改…...
Redux的基本使用,从入门到入土
目录 一、初步使用Redux 1.安装Redux 2.配置状态机 二、Redux的核心概念 1.工作流程 2.工作流程 三、优化Redux 1.对action进行优化 2.type常量 3.reducer优化 四、react-redux使用 1.安装react-redux 2.全局注入store仓库 3.组件关联仓库 五、状态机的Hook 1.u…...
GDOUCTF2023-部分re复现
目录 [GDOUCTF 2023]Check_Your_Luck [GDOUCTF 2023]Tea [GDOUCTF 2023]doublegame [GDOUCTF 2023]Check_Your_Luck 打开题目是一串代码,明显的z3约束器求解 直接上脚本 import z3 from z3 import Reals z3.Solver() vReal(v) xReal(x) yReal(y) wReal(w) zRea…...
Java学习17(IO模型详解)
1、何为IO? I/O(Input/Outpu) 即输入/输出 。 从计算机结构的角度来解读一下 I/O。 根据冯.诺依曼结构,计算机结构分为 5 大部分:运算器、控制器、存储器、输入设备、输出设备。 输入设备(比如键盘&am…...
Vue-全局过滤器以及进阶操作
前言 上篇文件讲述了,Vue全局过滤器的基本使用:Vue过滤器的基本使用 本篇将延续上文,讲述vue中过滤器的进阶操作 过滤器传参 如果有一天,多个地方使用过滤器,而且需要传递参数,那么可以这么写 多个过滤…...
财报解读:涅槃重生之后,新东方还想再造一个“文旅甄选”?
新东方逐渐走出了“微笑曲线”。 图源:新东方2023财年Q3财报 2023年4月19日,新东方披露了2023财年Q3财报(截至2023年2月28日止),营收7.5亿美元,同比增长22.8%;归母净利润为8165万美元ÿ…...
华为OD机试 - 过滤组合字符串(Python)
题目描述 每个数字关联多个字母,关联关系如下: 0 关联 “a”,”b”,”c” 1 关联 “d”,”e”,”f” 2 关联 “g”,”h”,”i” 3 关联 “j”,”k”,”l” 4 关联 “m”,”n”,”o” 5 关联 “p”,”q”,”r” 6 关联 “s”,”t” 7 关联 “u”,”v” 8 关联 “w”,”x” 9 …...
maven简单使用
实验课的作业用一大堆框架/库,统统要用maven管理。 头一次用,真痛苦。 所幸得以解决,maven真香~ 一步一步来。 1. maven 不是java人,只能说说粗浅的理解了。 简单来说,maven是一个管理项目的工具&…...
HTML学习笔记一
目录 HTML学习笔记 一、HTML标签 1、HTML语法规范 1.1标签的语法概述 1.2标签关系 2、HTML基本结构标签 2.1第一个HTML 2.2基本结构标签总结 3、开发工具 4、HTML常用标签 4.1标签的语义 4.2标题标签 4.3段落和换行标签 4.4文本格式化标签 4.5div和span标签 4.…...
人工智能十大流行算法,通俗易懂讲明白
人工智能是什么?很多人都知道,但大多又都说不清楚。 事实上,人工智能已经存在于我们生活中很久了。 比如我们常常用到的邮箱,其中垃圾邮件过滤就是依靠人工智能;比如每个智能手机都配备的指纹识别或人脸识别&#x…...
支持中英双语和多种插件的开源对话语言模型,160亿参数
一、开源项目简介 MOSS是一个支持中英双语和多种插件的开源对话语言模型,moss-moon系列模型具有160亿参数,在FP16精度下可在单张A100/A800或两张3090显卡运行,在INT4/8精度下可在单张3090显卡运行。MOSS基座语言模型在约七千亿中英文以及代码…...
SQL基础培训10-复杂查询原理
知识点: 1、SQL查询语句逻辑执行顺序 下面是一个查询语句的逻辑执行顺序(每段语句都标明了执行顺序号): 执行1:FROM 执行2:...
Vue记事本应用实现教程
文章目录 1. 项目介绍2. 开发环境准备3. 设计应用界面4. 创建Vue实例和数据模型5. 实现记事本功能5.1 添加新记事项5.2 删除记事项5.3 清空所有记事 6. 添加样式7. 功能扩展:显示创建时间8. 功能扩展:记事项搜索9. 完整代码10. Vue知识点解析10.1 数据绑…...
Python:操作 Excel 折叠
💖亲爱的技术爱好者们,热烈欢迎来到 Kant2048 的博客!我是 Thomas Kant,很开心能在CSDN上与你们相遇~💖 本博客的精华专栏: 【自动化测试】 【测试经验】 【人工智能】 【Python】 Python 操作 Excel 系列 读取单元格数据按行写入设置行高和列宽自动调整行高和列宽水平…...
【位运算】消失的两个数字(hard)
消失的两个数字(hard) 题⽬描述:解法(位运算):Java 算法代码:更简便代码 题⽬链接:⾯试题 17.19. 消失的两个数字 题⽬描述: 给定⼀个数组,包含从 1 到 N 所有…...
《用户共鸣指数(E)驱动品牌大模型种草:如何抢占大模型搜索结果情感高地》
在注意力分散、内容高度同质化的时代,情感连接已成为品牌破圈的关键通道。我们在服务大量品牌客户的过程中发现,消费者对内容的“有感”程度,正日益成为影响品牌传播效率与转化率的核心变量。在生成式AI驱动的内容生成与推荐环境中࿰…...
Vue2 第一节_Vue2上手_插值表达式{{}}_访问数据和修改数据_Vue开发者工具
文章目录 1.Vue2上手-如何创建一个Vue实例,进行初始化渲染2. 插值表达式{{}}3. 访问数据和修改数据4. vue响应式5. Vue开发者工具--方便调试 1.Vue2上手-如何创建一个Vue实例,进行初始化渲染 准备容器引包创建Vue实例 new Vue()指定配置项 ->渲染数据 准备一个容器,例如: …...
vue3 字体颜色设置的多种方式
在Vue 3中设置字体颜色可以通过多种方式实现,这取决于你是想在组件内部直接设置,还是在CSS/SCSS/LESS等样式文件中定义。以下是几种常见的方法: 1. 内联样式 你可以直接在模板中使用style绑定来设置字体颜色。 <template><div :s…...
Ascend NPU上适配Step-Audio模型
1 概述 1.1 简述 Step-Audio 是业界首个集语音理解与生成控制一体化的产品级开源实时语音对话系统,支持多语言对话(如 中文,英文,日语),语音情感(如 开心,悲伤)&#x…...
自然语言处理——Transformer
自然语言处理——Transformer 自注意力机制多头注意力机制Transformer 虽然循环神经网络可以对具有序列特性的数据非常有效,它能挖掘数据中的时序信息以及语义信息,但是它有一个很大的缺陷——很难并行化。 我们可以考虑用CNN来替代RNN,但是…...
【HarmonyOS 5 开发速记】如何获取用户信息(头像/昵称/手机号)
1.获取 authorizationCode: 2.利用 authorizationCode 获取 accessToken:文档中心 3.获取手机:文档中心 4.获取昵称头像:文档中心 首先创建 request 若要获取手机号,scope必填 phone,permissions 必填 …...
【VLNs篇】07:NavRL—在动态环境中学习安全飞行
项目内容论文标题NavRL: 在动态环境中学习安全飞行 (NavRL: Learning Safe Flight in Dynamic Environments)核心问题解决无人机在包含静态和动态障碍物的复杂环境中进行安全、高效自主导航的挑战,克服传统方法和现有强化学习方法的局限性。核心算法基于近端策略优化…...
