使用 ESP32 和 PlatformIO (arduino框架)实现 Over-the-Air(OTA)固件更新
使用 ESP32 和 PlatformIO 实现 Over-the-Air(OTA)固件更新
摘要:
本文将介绍如何在 ESP32 上使用 PlatformIO 环境实现 OTA(Over-the-Air)固件更新。OTA 更新使得在设备部署在远程位置时,无需物理接触设备,就可以通过网络更新固件,大大提高了设备维护和管理的便捷性。
介绍:
随着物联网技术的发展,越来越多的设备需要进行固件更新以修复漏洞、添加新功能或提高性能。传统的固件更新方式需要通过串口连接或者直接物理接触设备,但是当设备分布在远程位置时,这种方式就显得非常不便。而 OTA 固件更新技术则能够解决这个问题,使得固件更新可以通过网络实现。
工程配置:
创建新分区表: 在项目目录下创建.cvs后缀文件,文件名随意,这里我以partition.csv文件名为例

打开partition.csv将下面的分区配置,直接粘贴上去
# Name Type SubType Offset Size Flags
nvs, data, nvs, 0x9000, 0x5000
otadata,data, ota, 0xe000, 0x2000
app0, app, ota_0, 0x10000, 0x140000
app1, app, ota_1, 0x150000, 0x140000
spiffs, data, spiffs, 0x290000, 0x170000
配置 PlatformIO 项目: 在项目的 platformio.ini 文件中添加下面这行 ,配置选用自定义分区表partition.csv改为你自己创建的分区表
board_build.partitions = partition.csv
Update库
Update 库是 Arduino 核心库中用于 OTA(Over-The-Air)固件更新的一个关键库。它提供了用于在 ESP8266 和 ESP32 上进行固件更新的一组功能和类。
主要功能:
-
OTA 固件更新: Update 库允许你通过 WiFi 网络进行固件更新,而无需物理连接到设备。这使得你可以远程更新设备上的固件,而不必重新连接电脑或使用串口进行更新。
-
简单易用的接口: Update 库提供了一组简单易用的函数和类,用于初始化 OTA 更新功能、写入固件数据和结束 OTA 更新过程。这些函数和类的接口设计得很简洁,使得在代码中集成 OTA 更新功能变得非常容易。
-
OTA 更新错误处理: Update 库还提供了一些用于处理 OTA 更新过程中可能出现的错误的函数。例如,你可以使用
hasError()函数检查更新过程是否出现了错误,并使用printError()函数打印出错误信息以进行调试。
主要类和函数:
-
Update 类: Update 类是 Update 库的核心部分,提供了用于 OTA 固件更新的主要功能。它包含了
begin()、write()、end()等函数,用于初始化更新、写入固件数据和结束更新过程。 -
begin() 函数: 这个函数用于初始化 OTA 更新功能,并可以指定固件的大小,也可以选择不指定大小,然后在上传固件时自动检测大小。
-
write() 函数: 这个函数用于将接收到的固件数据写入到更新对象中。通常在上传固件的过程中调用,用于将固件的每个数据块写入到更新对象中。
-
end() 函数: 这个函数用于结束 OTA 更新过程。你可以选择设置固件大小为当前接收到的数据大小,也可以选择不设置大小,以最后接收到的数据大小为准。
-
hasError() 函数: 这个函数用于检查 OTA 更新过程中是否发生了错误。如果在更新过程中出现了错误,则返回 true;否则返回 false。
-
printError() 函数: 这个函数用于打印出 OTA 更新过程中发生的错误信息。通常在发生错误时使用,以便调试和排查问题。
使用 Arduino 的 Update 库进行 OTA(Over-The-Air)固件更新非常简单,下面我将逐步说明基本的使用方法。
步骤 1:包含头文件和声明全局变量
首先,在你的 Arduino 项目中包含 Update 库的头文件,并声明 WiFi 相关的全局变量,例如 WiFi SSID 和密码。
#include <WiFi.h>
#include <Update.h>const char* ssid = "YourSSID";
const char* password = "YourPassword";
步骤 2:连接到 WiFi 网络
在 setup() 函数中,连接到你的 WiFi 网络。
void setup() {Serial.begin(115200);WiFi.begin(ssid, password);while (WiFi.status() != WL_CONNECTED) {delay(1000);Serial.println("Connecting to WiFi...");}Serial.println("Connected to WiFi!");
}
步骤 3:初始化 OTA 更新
在 setup() 函数中,初始化 OTA 更新功能。
void setup() {// 连接到 WiFi 网络(省略)// 初始化 OTA 更新if (Update.begin()) {Serial.println("OTA update begin...");// 在这里写入固件数据// 结束 OTA 更新过程Update.end();Serial.println("OTA update complete!");} else {Serial.println("OTA update failed!");}
}
步骤 4:写入固件数据
在初始化 OTA 更新后,你可以通过 Update.write() 函数将固件数据写入更新对象。你可以在这里接收固件数据的回调函数中调用此函数。
void setup() {// 连接到 WiFi 网络(省略)// 初始化 OTA 更新(省略)// 接收固件数据的回调函数server.on("/update", HTTP_POST, []() {// 结束 OTA 更新过程Update.end(true);}, []() {HTTPUpload& upload = server.upload();if (upload.status == UPLOAD_FILE_START) {if (!Update.begin(UPDATE_SIZE_UNKNOWN)) {Update.printError(Serial);}} else if (upload.status == UPLOAD_FILE_WRITE) {if (Update.write(upload.buf, upload.currentSize) != upload.currentSize) {Update.printError(Serial);}} else if (upload.status == UPLOAD_FILE_END) {if (Update.end(true)) {Serial.println("OTA update complete!");} else {Update.printError(Serial);}}});// 开启 Web 服务器(省略)
}
步骤 5:结束 OTA 更新
最后,在你写入完所有固件数据后,调用 Update.end() 函数结束 OTA 更新过程。
void setup() {// 连接到 WiFi 网络(省略)// 初始化 OTA 更新(省略)// 写入固件数据(省略)// 结束 OTA 更新Update.end(true);
}
项目案例
这里我为大家提供了一个项目案例,这个项目实现了一个简单的ESP32固件OTA(Over-The-Air)更新方法,大家稍作修改就可以直接移植到你自己的项目中,它利用ESP32的WiFi功能搭建了个小AP热点,让我们通过一个简单的Web界面上传新的固件文件,然后用Arduino的Update库自动更新固件。通过这个项目,我们可以方便地在没有外部网络连接的情况下,通过无线方式更新ESP32设备的固件。
案例代码
#include <Arduino.h>
#include <TFT_eSPI.h>
#include <math.h>
#include <WiFi.h>
#include <WebServer.h>
#include <Update.h>
// #include <ArduinoMDNS.h> // 引入mDNS库const char* ssid = "ESP32-c3_OTAdemo"; // AP的名称
const char* password = "123456789"; // AP的密码IPAddress local_IP(192, 168, 4, 1); // 静态IP地址
IPAddress gateway(192, 168, 4, 1); // 网关IP地址WebServer server(80);TFT_eSPI tft;const char* updateIndex ="<html>""<head>""<meta charset=\"UTF-8\">""<title>ESP32 OTA 更新</title>""<style>""body { font-family: Arial, sans-serif; text-align: center; }""h1 { color: #333; }""form { margin-top: 20px; }""input[type=file] { display: block; margin: 20px auto; }""input[type=submit] { margin-top: 20px; padding: 10px 20px; font-size: 18px; }""</style>""</head>""<body>""<h1>欢迎使用 ESP32 OTA 更新</h1>""<form method='POST' action='/update' enctype='multipart/form-data'>""<input type='file' name='update' accept='.bin'>""<input type='submit' value='上传固件'>""</form>""</body>""</html>";void handleRoot() {server.sendHeader("Location", "/update");server.send(302, "text/plain", "");
}void setup() {Serial.begin(115200); // 初始化串口,波特率为115200Serial.println("Booting...");// 将 ESP32 设置为 AP 模式并指定静态 IP 地址WiFi.softAP(ssid, password);WiFi.softAPConfig(local_IP, gateway, IPAddress(255, 255, 255, 0));Serial.print("Access Point IP address: ");Serial.println(WiFi.softAPIP()); // 打印 ESP32 的 AP IP 地址// 设置服务器处理函数server.on("/", HTTP_GET, handleRoot); // 根路由重定向到 OTA 页面server.on("/update", HTTP_GET, []() {server.sendHeader("Connection", "close");server.send(200, "text/html", updateIndex);});server.on("/update", HTTP_POST, []() {server.sendHeader("Connection", "close");//动态显示结果String message = Update.hasError() ? "更新失败" : "更新成功。重新启动…";server.sendHeader("Content-Type", "text/html; charset=utf-8");server.send(200, "text/html", "<span style='font-size: 24px;'>" + message + "</span>");delay(1000);ESP.restart();}, []() {HTTPUpload& upload = server.upload(); //用于处理上传的文件数据if (upload.status == UPLOAD_FILE_START) {Serial.printf("Update: %s\n", upload.filename.c_str());if (!Update.begin(UPDATE_SIZE_UNKNOWN)) { // 以最大可用大小开始Update.printError(Serial);}} else if (upload.status == UPLOAD_FILE_WRITE) {// 将接收到的数据写入Update对象if (Update.write(upload.buf, upload.currentSize) != upload.currentSize) {Update.printError(Serial);}} else if (upload.status == UPLOAD_FILE_END) {if (Update.end(true)) { // 设置大小为当前大小Serial.printf("Update Success: %u bytes\n", upload.totalSize);} else {Update.printError(Serial);}}});server.begin();Serial.println("HTTP server started");//程序逻辑 版本1.1Serial.println();Serial.println("NEW ESP32C3!!");tft.begin();tft.setRotation(3);tft.setTextFont(2);tft.fillScreen(TFT_BLACK);tft.drawString("NEW ESP32C3!!", 0, 0);tft.drawRect(2, 20, 100, 20, TFT_BROWN);
}void loop() {server.handleClient();}
获取固件文件
PlatformIO 本身并不提供直接导出固件文件的功能,但你可以在 PlatformIO 中构建项目,并手动在构建目录中找到生成的固件文件。
一般情况下,PlatformIO 会将编译生成的固件文件放置在项目的 .pio 目录中。具体来说,对于 ESP32 来说,固件文件通常位于 .pio/build/<board_name>/firmware.bin,其中 <board_name> 是你的开发板名称,例如 esp32dev。

OTA操作
程序烧录到开发板之后,在电脑端连接wifi ESP32-c3_OTAdemo,密码为123456789

连接上WIFI之后,打开浏览器输入URL
http://192.168.4.1/update

下图即是我们的OTA上传页面

点击选择文件

选择需要下载的固件文件.bin文件

点击上传固件

如果上传成功,显示更新成功,并且重新启动,如果失败,显示更新失败。

总结:
通过使用 PlatformIO 环境和 ESP32 开发板,我们可以轻松实现 OTA 固件更新功能。这使得固件更新变得更加灵活和便捷,大大提高了设备管理的效率。在物联网应用中,OTA 技术将会发挥越来越重要的作用,帮助我们更好地维护和管理设备。
相关文章:
使用 ESP32 和 PlatformIO (arduino框架)实现 Over-the-Air(OTA)固件更新
使用 ESP32 和 PlatformIO 实现 Over-the-Air(OTA)固件更新 摘要: 本文将介绍如何在 ESP32 上使用 PlatformIO 环境实现 OTA(Over-the-Air)固件更新。OTA 更新使得在设备部署在远程位置时,无需物理接触设…...
学习笔记——路由网络基础——汇总静态路由
4、汇总静态路由 (1)定义 静态路由汇总:多条静态路由都使用相同的送出接口或下一跳 IP 地址。(将多条路由汇总成一条路由表示) (2)目的 1.减少路由条目数量,减小路由表,加快查表速度 2.增加网络稳定性 (3)路由黑洞以及路由环路的产生…...
这10个python库,下载都超过5亿
python的库数不胜数。哪些库使用得最多呢。今天分享10个下载都超过5亿的python库。从高到低排序 第一名:Urllib3 下载次数:8.93亿次 介绍:Urllib3是一个功能强大且用户友好的HTTP客户端库,提供了许多Python标准库中没有的特性&…...
Vue3【十一】08使用toRefs和toRef
08使用toRefs和toRef toRefs()函数将person对象中的name和age属性转换为响应式引用,并返回一个对象,对象中的name和age属性都是响应式引用,具有响应式功能。 toRef()函数将person对象中的name属性转换为响应式引用,并返回一个响应…...
离散数学---树
目录 1.基本概念及其相关运用 2.生成树 3.有向树 4.最优树 5.前缀码 1.基本概念及其相关运用 (1)无向树:连通而且没有回路的无向图就是无向树; 森林就是有多个连通分支,每个连通分支都是树的无连通的无向图&…...
【栈】1106. 解析布尔表达式
本文涉及知识点 栈 LeetCode 1106. 解析布尔表达式 布尔表达式 是计算结果不是 true 就是 false 的表达式。有效的表达式需遵循以下约定: ‘t’,运算结果为 true ‘f’,运算结果为 false ‘!(subExpr)’,运算过程为对内部表达式…...
u盘内容无故消失了是什么原因?u盘部分内容无故消失了怎么恢复
在数字化时代,U盘作为便携存储设备,承载着许多重要的数据。然而,有时我们可能会遭遇U盘部分内容无故消失的情况,这无疑给我们的工作和生活带来了不小的困扰。本文将为您解析U盘内容消失的可能原因,并分享几招实用的数据…...
glm-4v-9b 部署
glm-4v-9b 模型文件地址 GLM-4 仓库文件地址 官方测试 硬件配置和系统要求 官方测试硬件信息: OS: Ubuntu 22.04Memory: 512G…...
Ansible——unarchive模块
目录 参数总结 基础语法 常见的命令行示例 示例1:解压缩文件到指定目录 示例2:解压缩文件并设置权限 示例3:远程URL解压缩 示例4:强制覆盖现有文件 具体步骤和示例 示例5:只要文件解压后,如果存在…...
Ansible——get_url模块
目录 主要用途 参数总结 基本语法示例 使用示例 示例1:下载文件 示例2:使用校验和验证文件 示例3:使用 HTTP 基本认证 示例4:通过代理服务器下载文件 示例5:设置文件权限、所有者和组 示例6:强制…...
macbook本地部署 pyhive环境连接 hive用例
前言 公司的测试和生产环境中尚未提供基于Hive的客户端。若希望尝试操作Hive表,目前一个可行的方案是使用Python语言,通过借助pyhive库,您可以对Hive表进行各种操作。以下是一些示例记录供您参考。 一、pyhive是什么? PyHive是一…...
物理安全防护如何创新强化信息安全体系?
物理安全防护是信息安全体系的重要组成部分,它通过保护实体设施、设备和介质等,防止未授权访问、破坏、盗窃等行为,从而为信息系统提供基础的安全保障。要创新强化信息安全体系中的物理安全防护,可以从以下几个方面着手࿱…...
【JAVASE】日期与时间类(上)
一:概述 从JAVA SE 8开始提供了java.time包,该包中有专门处理日期和时间的类。 LocalDate LocalDateTime 和LocalTime 类的对象封装和日期、时间有关的数据,这三个类都是final类,而且不提供修改数据的方法,即这…...
如果需要精确的答案,请避免使用float和double
float和double主要为了科学计算和工程计算而设计,执行二进制浮点运算,这是为了在广泛的数值范围上提供较为精确的快速近似计算而精心设计的。然而,它们没有提供完全精确的结果,所以不适合用于需要精确结果的场合,尤其是…...
大模型,也在卷价格
“百模大战”已从算力战、规模战蔓延到了价格战。 5月15日,字节跳动宣布豆包主力模型(小于等于32K)在企业市场的定价只有0.0008元/千Tokens,0.8厘就能处理1500多个汉字,比行业便宜99.3%;5月21日࿰…...
开关电源中电感设计
开关电源设计中电感 只有充分理解电感在DC/DC电路中发挥的作用,才能更优的设计DC/DC电路。本文还包括对同步DC/DC及异步DC/DC概念的解释。 在开关电源的设计中电感的设计为工程师带来的许多的挑战。工程师不仅要选择电感值,还要考虑电感可承受的电流,绕线电阻,机械尺寸等…...
机器视觉——硬件常用基础知识
光源 机器视觉中光源的作用 1)强化特征,弱化背景 2)光源打得好,图好了,后期算法更简化 3)图好了,测试速度更高 各种光源的综合性能对比及为啥使用LED灯 光的颜色的选择 白色光:通常用…...
宝塔 php7.4 安装SQLserver扩展
一、加入微软源 curl https://packages.microsoft.com/config/rhel/7/prod.repo > /etc/yum.repos.d/mssqlrelease.repo二、安装odbc驱动程序 yum install msodbcsql mssql-tools unixODBC-devel 三、安装php7.4对应的pdo_sqlsrv扩展包 # 下载 wget http://pecl.php.net/…...
C++中的常见I/O方式
目录 摘要 1. 标准输入输出(Standard I/O) 2. 文件输入输出(File I/O) 3. 字符串流(String Stream) 4. 低级文件I/O(Low-level File I/O) 5. 内存映射文件(Memory-Mapped File I/O) 6. 网络I/O(Network I/O) 服务器端 客户端 摘要 C++中的输入输出操作(…...
Java Web学习笔记23——Vue项目简介
Vue项目简介: Vue项目-创建: 命令行:vue create vue-project01 图形化界面:vue ui 在命令行中切换到项目文件夹中,然后执行vue ui命令。 只需要路由功能。这个路由功能,开始不是很理解。 创建项目部保存…...
智慧教育——解读AI一体化智慧校园解决方案【附全文阅读】
适应人群为学校管理人员、教师、学生、技术运维人员及教育信息化建设相关从业者。主要内容围绕 AI 一体化智慧校园建设,阐述总体规划及革命性意义(提升教学管理水平、降低成本等);介绍八大应用中心(教学管理、物联网管控、校园安全等),涵盖智能选课排课、校园安防监控等…...
嵌入式开发者必看:GitHub高星项目实战解析
1. 嵌入式开发者不可错过的GitHub高星项目盘点作为一名在嵌入式领域摸爬滚打多年的开发者,我深知优质开源项目对技术成长的重要性。GitHub这个宝藏平台上其实藏着不少嵌入式相关的精品项目,今天我就带大家深度剖析几个值得研究的项目,并分享我…...
2025届必备的六大降重复率助手横评
Ai论文网站排名(开题报告、文献综述、降aigc率、降重综合对比) TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 纵然人工智能辅助毕业论文写作现如今已然成为一种学术方面的新常态,可是却需要去…...
2025届必备的六大降重复率平台横评
Ai论文网站排名(开题报告、文献综述、降aigc率、降重综合对比) TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 在内容创作范畴当中,要是打算削减 AIGC 特性,那就得从语言风格、逻辑…...
JavaScript高频八股
一、原型和原型链1、概念:每个对象都有一个隐藏的属性 __proto__(原型),指向它创建时的构造函数的 prototype(原型对象)。当访问对象的一个属性或方法时,如果对象本身没有,就会去它的…...
终极指南:3步快速修复Visual C++运行库,让Windows告别DLL错误
终极指南:3步快速修复Visual C运行库,让Windows告别DLL错误 【免费下载链接】vcredist AIO Repack for latest Microsoft Visual C Redistributable Runtimes 项目地址: https://gitcode.com/gh_mirrors/vc/vcredist Visual C运行库是Windows系统…...
光学工程师进阶指南:从入门到精通的实战路径
1. 光学工程师的职业发展路径 光学工程师的成长就像搭积木,需要从最基础的模块开始,一层层往上搭建。我刚入行时也走过不少弯路,后来才明白这个职业的发展是有明确路径的。一般来说,我们可以把成长过程分为三个阶段:初…...
OpenMMLab 环境配置实战:从 YOLO 项目报错到模块化开发的避坑指南
1. 从YOLO项目报错说起:OpenMMLab环境配置的典型痛点 最近在复现一个基于YOLOv5改进的OpenMMLab项目时,遇到了让人头疼的ModuleNotFoundError: No module named mmdet报错。这个场景太典型了——明明项目目录里清清楚楚躺着mmdet文件夹,Pytho…...
脉冲注入法与电感法无刷电机BLDC控制器方案
脉冲注入法,持续注入,启动低速运行过程中注入,电感法,ipd,力矩保持,无霍尔无感方案,媲美有霍尔效果。 bldc控制器方案,无刷电机。 提供源码,原理图。一、文档引言 本文基…...
Linux内核中的锁机制对比:选择合适的同步原语
Linux内核中的锁机制对比:选择合适的同步原语 作为一名深耕操作系统和嵌入式开发的工程师,我对Linux内核中的各种锁机制有着深入的理解。不同的锁适用于不同的场景,选择合适的锁对于系统性能至关重要。 内核锁的类型 1. 互斥锁(Mu…...
