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

ESP32 网络计时器,包含自动保存

简介

本代码是基于ESP32开发板实现的一个计时器功能,具备倒计时、计时器时长选择、显示当前时间、有源蜂鸣器报警等功能。代码中使用了WiFi网络连接、NTP时间同步、EEPROM存储等功能。通过按钮控制计时器的开始、停止和计时器时长的选择。

运行原理概述

在ESP32开发板上,使用了三个按钮,分别为开始计时按钮(BUTTON1)、停止计时按钮(BUTTON2)和计时器时长选择按钮。首先连接WiFi网络,同步NTP时间,并初始化OLED屏幕。

然后通过setdefaulttime()函数设置计时器默认时长,并初始化按钮引脚。

在主循环中通过checkTimerButtons()函数检测按钮状态,若按钮被按下,根据按钮类型执行相应的操作。如果开始计时按钮被按下,则开始计时,并在OLED屏幕上显示计时器倒计时时间。

如果停止计时按钮被按下,则停止计时,并根据按钮状态执行相应的操作。如果计时器时长选择按钮被按下,则切换计时器时长,并在OLED屏幕上显示新的计时器时长。

特色功能

1.具备倒计时功能,可以按照设定的计时器时长进行倒计时,到时后有报警提示。

2.计时器时长可以选择,用户可以根据需求选择不同的计时器时长。

3.通过WiFi网络连接和NTP时间同步功能,保证了计时器的时间准确性。

4.使用EEPROM存储功能,保存计时器时长选择的状态,即使重新上电,也能保留上一次的时长选择状态。

重要函数的解释

  1. void setdefaulttime()函数:设置计时器默认时长,并从EEPROM中读取上一次的时长选择状态。

  2. void checkTimerButtons()函数:检测按钮状态,并根据按钮类型执行相应的操作。

  3. void updateTimeDisplay()函数:更新OLED屏幕上的时间和计时器倒计时时间。

  4. bool saveIntToEEPROM(int value, int address)函数:将整数值存储到EEPROM中。

  5. int readIntFromEEPROM(int address)函数:从EEPROM中读取整数值。

ESP32开发板引脚硬件设备引脚连接方式
GPIO21OLED_SDAI2C数据线
GPIO22OLED_SCLI2C时钟线
GPIO35BUTTON1按钮输入
GPIO34BUTTON2按钮输入
GPIO32BUZZER_PIN有源蜂鸣器引脚
3.3VOLED_VCCOLED显示屏电源
GNDOLED_GNDOLED显示屏电源和地
3.3VBUTTON1按钮电源
3.3VBUTTON2按钮电源
GNDBUTTON1按钮电源和地
GNDBUTTON2按钮电源和地

#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开发板实现的一个计时器功能&#xff0c;具备倒计时、计时器时长选择、显示当前时间、有源蜂鸣器报警等功能。代码中使用了WiFi网络连接、NTP时间同步、EEPROM存储等功能。通过按钮控制计时器的开始、停止和计时器时长的选择。 运行原理概述 在ESP32开…...

【ChatGPT】阿里版 ChatGPT 突然官宣意味着什么?

Yan-英杰的主页 悟已往之不谏 知来者之可追 C程序员&#xff0c;2024届电子信息研究生 目录 阿里版 ChatGPT 突然官宣 ​ ChatGPT 技术在 AI 领域的重要性 自然语言生成 上下文连续性 多语言支持 ChatGPT 未来可能的应用场景 社交领域 商业领域 ​编辑 医疗领域…...

IPEmotion控制模块-PID循环应用

IPEmotion专业版、开发版支持控制模块&#xff0c;并且该模块支持函数发生器、PID控制器、路由器、序列控制和序列控制块以及参考曲线生成器。本文主要针对PID&#xff08;P&#xff1a;Proportional control 比例控制&#xff1b;I&#xff1a;Integral control 积分控制&…...

【元分析研究方法】学习笔记2.检索文献(含100种学术文献搜索清单链接)

检索文献 该步骤的作用该步骤中需要注意的问题该步骤中部分知识点我的收获 参考来源&#xff1a;库珀 (Cooper, H. M. )., 李超平, & 张昱城. (2020). 元分析研究方法: A step-by step approach. 中国人民大学出版社. 该步骤的作用 1.识别相关文献的来源&#xff1b; 2.识别…...

题目:16版.自由落体

1、实验要求 本实验要求&#xff1a;模拟物体从10000米高空掉落后的反弹行为。 1-1. 创建工程并配置环境&#xff1a; 1-1.1. 限制1. 工程取名&#xff1a;SE_JAVA_EXP_E009。 1-1.2. 限制2. 创建包&#xff0c;取名&#xff1a;cn.campsg.java.experiment。 1-1.3. 限制3. 创建…...

视频可视化搭建项目,通过简单拖拽方式快速生产一个短视频

一、开源项目简介 《视搭》是一个视频可视化搭建项目。您可以通过简单的拖拽方式快速生产一个短视频&#xff0c;使用方式就像易企秀或百度 H5 等 h5 搭建工具一样的简单。目前行业内罕有关于视频可视化搭建的开源项目&#xff0c;《视搭》是一个相对比较完整的开源项目&#…...

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&#xff1a;data、from、tonetwork->linkiplink source en…...

计算机网络笔记(横向)

该笔记也是我考研期间做的整理。一般网上的笔记是按照章节纪录的&#xff0c;我是按照知识点分类纪录的&#xff0c;大纲如下&#xff1a; 文章目录 1. 各报文1.1 各报文头部详解1.2 相关口诀 2. 各协议2.1 各应用层协议使用的传输层协议与端口2.2 各协议的过程2.2.1 数据链路层…...

0.redis-实践

1.redis内存设置多少,默认是0&#xff0c;不限制 2.如何配置&#xff0c;修改内存大小 1) 查看最大占用内存 # maxmeory <bytes> 或者 config get maxmemory 2) 默认内存多少可以用: 64位系统下不限制&#xff0c;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 打开题目是一串代码&#xff0c;明显的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&#xff08;Input/Outpu&#xff09; 即输入&#xff0f;输出 。 从计算机结构的角度来解读一下 I/O。 根据冯.诺依曼结构&#xff0c;计算机结构分为 5 大部分&#xff1a;运算器、控制器、存储器、输入设备、输出设备。 输入设备&#xff08;比如键盘&am…...

Vue-全局过滤器以及进阶操作

前言 上篇文件讲述了&#xff0c;Vue全局过滤器的基本使用&#xff1a;Vue过滤器的基本使用 本篇将延续上文&#xff0c;讲述vue中过滤器的进阶操作 过滤器传参 如果有一天&#xff0c;多个地方使用过滤器&#xff0c;而且需要传递参数&#xff0c;那么可以这么写 多个过滤…...

财报解读:涅槃重生之后,新东方还想再造一个“文旅甄选”?

新东方逐渐走出了“微笑曲线”。 图源&#xff1a;新东方2023财年Q3财报 2023年4月19日&#xff0c;新东方披露了2023财年Q3财报&#xff08;截至2023年2月28日止&#xff09;&#xff0c;营收7.5亿美元&#xff0c;同比增长22.8%&#xff1b;归母净利润为8165万美元&#xff…...

华为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简单使用

实验课的作业用一大堆框架/库&#xff0c;统统要用maven管理。 头一次用&#xff0c;真痛苦。 所幸得以解决&#xff0c;maven真香&#xff5e; 一步一步来。 1. maven 不是java人&#xff0c;只能说说粗浅的理解了。 简单来说&#xff0c;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.…...

人工智能十大流行算法,通俗易懂讲明白

人工智能是什么&#xff1f;很多人都知道&#xff0c;但大多又都说不清楚。 事实上&#xff0c;人工智能已经存在于我们生活中很久了。 比如我们常常用到的邮箱&#xff0c;其中垃圾邮件过滤就是依靠人工智能&#xff1b;比如每个智能手机都配备的指纹识别或人脸识别&#x…...

支持中英双语和多种插件的开源对话语言模型,160亿参数

一、开源项目简介 MOSS是一个支持中英双语和多种插件的开源对话语言模型&#xff0c;moss-moon系列模型具有160亿参数&#xff0c;在FP16精度下可在单张A100/A800或两张3090显卡运行&#xff0c;在INT4/8精度下可在单张3090显卡运行。MOSS基座语言模型在约七千亿中英文以及代码…...

SQL基础培训10-复杂查询原理

知识点: 1、SQL查询语句逻辑执行顺序 下面是一个查询语句的逻辑执行顺序(每段语句都标明了执行顺序号): 执行1:FROM 执行2:...

VB.net复制Ntag213卡写入UID

本示例使用的发卡器&#xff1a;https://item.taobao.com/item.htm?ftt&id615391857885 一、读取旧Ntag卡的UID和数据 Private Sub Button15_Click(sender As Object, e As EventArgs) Handles Button15.Click轻松读卡技术支持:网站:Dim i, j As IntegerDim cardidhex, …...

Swift 协议扩展精进之路:解决 CoreData 托管实体子类的类型不匹配问题(下)

概述 在 Swift 开发语言中&#xff0c;各位秃头小码农们可以充分利用语法本身所带来的便利去劈荆斩棘。我们还可以恣意利用泛型、协议关联类型和协议扩展来进一步简化和优化我们复杂的代码需求。 不过&#xff0c;在涉及到多个子类派生于基类进行多态模拟的场景下&#xff0c;…...

MySQL中【正则表达式】用法

MySQL 中正则表达式通过 REGEXP 或 RLIKE 操作符实现&#xff08;两者等价&#xff09;&#xff0c;用于在 WHERE 子句中进行复杂的字符串模式匹配。以下是核心用法和示例&#xff1a; 一、基础语法 SELECT column_name FROM table_name WHERE column_name REGEXP pattern; …...

华为云Flexus+DeepSeek征文|DeepSeek-V3/R1 商用服务开通全流程与本地部署搭建

华为云FlexusDeepSeek征文&#xff5c;DeepSeek-V3/R1 商用服务开通全流程与本地部署搭建 前言 如今大模型其性能出色&#xff0c;华为云 ModelArts Studio_MaaS大模型即服务平台华为云内置了大模型&#xff0c;能助力我们轻松驾驭 DeepSeek-V3/R1&#xff0c;本文中将分享如何…...

基于TurtleBot3在Gazebo地图实现机器人远程控制

1. TurtleBot3环境配置 # 下载TurtleBot3核心包 mkdir -p ~/catkin_ws/src cd ~/catkin_ws/src git clone -b noetic-devel https://github.com/ROBOTIS-GIT/turtlebot3.git git clone -b noetic https://github.com/ROBOTIS-GIT/turtlebot3_msgs.git git clone -b noetic-dev…...

Mysql中select查询语句的执行过程

目录 1、介绍 1.1、组件介绍 1.2、Sql执行顺序 2、执行流程 2.1. 连接与认证 2.2. 查询缓存 2.3. 语法解析&#xff08;Parser&#xff09; 2.4、执行sql 1. 预处理&#xff08;Preprocessor&#xff09; 2. 查询优化器&#xff08;Optimizer&#xff09; 3. 执行器…...

Xela矩阵三轴触觉传感器的工作原理解析与应用场景

Xela矩阵三轴触觉传感器通过先进技术模拟人类触觉感知&#xff0c;帮助设备实现精确的力测量与位移监测。其核心功能基于磁性三维力测量与空间位移测量&#xff0c;能够捕捉多维触觉信息。该传感器的设计不仅提升了触觉感知的精度&#xff0c;还为机器人、医疗设备和制造业的智…...

Python网页自动化Selenium中文文档

1. 安装 1.1. 安装 Selenium Python bindings 提供了一个简单的API&#xff0c;让你使用Selenium WebDriver来编写功能/校验测试。 通过Selenium Python的API&#xff0c;你可以非常直观的使用Selenium WebDriver的所有功能。 Selenium Python bindings 使用非常简洁方便的A…...

VisualXML全新升级 | 新增数据库编辑功能

VisualXML是一个功能强大的网络总线设计工具&#xff0c;专注于简化汽车电子系统中复杂的网络数据设计操作。它支持多种主流总线网络格式的数据编辑&#xff08;如DBC、LDF、ARXML、HEX等&#xff09;&#xff0c;并能够基于Excel表格的方式生成和转换多种数据库文件。由此&…...

解析“道作为序位生成器”的核心原理

解析“道作为序位生成器”的核心原理 以下完整展开道函数的零点调控机制&#xff0c;重点解析"道作为序位生成器"的核心原理与实现框架&#xff1a; 一、道函数的零点调控机制 1. 道作为序位生成器 道在认知坐标系$(x_{\text{物}}, y_{\text{意}}, z_{\text{文}}…...