0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

ESP32通过OTA无线局域网远程升级下载程序

jf_88434166 ? 来源:jf_88434166 ? 作者:jf_88434166 ? 2025-07-22 13:21 ? 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

概要

ESP32 的 OTA(Over-The-Air)升级功能允许通过无线网络Wi-Fi远程更新设备固件程序,无需物理连接(如 USB/UART)。比如ESP32设备在机壳内部不好拆卸,人离设备比较远,不好接线的环境可用到OTA方式给ESP32设备下载程序。为实现此功能每次更新程序的时候都要在代码里面加入OTA的相关代码。

远程OTA.jpg

创建ESP32-OTA无线局域网网页版远程升级入口

打开Arduino IDE中自带ESP32—OTAWebUpdater示例
在这里插入图片描述

#include < WiFi.h >
#include < WiFiClient.h >
#include < WebServer.h >
#include < ESPmDNS.h >
#include < Update.h >

const char* host = "esp32";
const char* ssid = "YXDZ1";
const char* password = "YXDZ#2023#1";

WebServer server(80);

/*
 * Login page
 */

const char* loginIndex =
 "< form name='loginForm' >"
    "< table width='20%' bgcolor='A09F9F' align='center' >"
        "< tr >"
            "< td colspan=2 >"
                "< center >< font size=4 >ESP32 Login Page< /font >< /center >"
                "
"
            "< /td >"
            "
"
            "
"
        "< /tr >"
        "< tr >"
             "< td >Username:< /td >"
             "< td >< input type='text' size=25 name='userid' >
< /td >"
        "< /tr >"
        "
"
        "
"
        "< tr >"
            "< td >Password:< /td >"
            "< td >< input type='Password' size=25 name='pwd' >
< /td >"
            "
"
            "
"
        "< /tr >"
        "< tr >"
            "< td >< input type='submit' onclick='check(this.form)' value='Login' >< /td >"
        "< /tr >"
    "< /table >"
"< /form >"
"< script >"
    "function check(form)"
    "{"
    "if(form.userid.value=='ESP32-OTA' && form.pwd.value=='ESP32-OTA')"
    "{"
    "window.open('/serverIndex')"
    "}"
    "else"
    "{"
    " alert('Error Password or Username')/*displays error message*/"
    "}"
    "}"
"< /script >";

/*
 * Server Index Page
 */

const char* serverIndex =
"< script src='/images/chaijie_default.png' >< /script >"
"< form method='POST' action='#' enctype='multipart/form-data' id='upload_form' >"
   "< input type='file' name='update' >"
        "< input type='submit' value='Update' >"
    "< /form >"
 "progress: 0%"
 "< script >"
  "$('form').submit(function(e){"
  "e.preventDefault();"
  "var form = $('#upload_form')[0];"
  "var data = new FormData(form);"
  " $.ajax({"
  "url: '/update',"
  "type: 'POST',"
  "data: data,"
  "contentType: false,"
  "processData:false,"
  "xhr: function() {"
  "var xhr = new window.XMLHttpRequest();"
  "xhr.upload.addEventListener('progress', function(evt) {"
  "if (evt.lengthComputable) {"
  "var per = evt.loaded / evt.total;"
  "$('#prg').html('progress: ' + Math.round(per*100) + '%');"
  "}"
  "}, false);"
  "return xhr;"
  "},"
  "success:function(d, s) {"
  "console.log('success!')"
 "},"
 "error: function (a, b, c) {"
 "}"
 "});"
 "});"
 "< /script >";

/*
 * setup function
 */
void setup(void) {
  Serial.begin(115200);

  // Connect to WiFi network
  WiFi.begin(ssid, password);
  Serial.println("");

  // Wait for connection
  while (WiFi.status() != WL_CONNECTED) {
    delay(500);
    Serial.print(".");
  }
  Serial.println("");
  Serial.print("Connected to ");
  Serial.println(ssid);
  Serial.print("IP address: ");
  Serial.println(WiFi.localIP());

  /*use mdns for host name resolution*/
  if (!MDNS.begin(host)) { //http://esp32.local
    Serial.println("Error setting up MDNS responder!");
    while (1) {
      delay(1000);
    }
  }
  Serial.println("mDNS responder started");
  /*return index page which is stored in serverIndex */
  server.on("/", HTTP_GET, []() {
    server.sendHeader("Connection", "close");
    server.send(200, "text/html", loginIndex);
  });
  server.on("/serverIndex", HTTP_GET, []() {
    server.sendHeader("Connection", "close");
    server.send(200, "text/html", serverIndex);
  });
  /*handling uploading firmware file */
  server.on("/update", HTTP_POST, []() {
    server.sendHeader("Connection", "close");
    server.send(200, "text/plain", (Update.hasError()) ? "FAIL" : "OK");
    ESP.restart();
  }, []() {
    HTTPUpload& upload = server.upload();
    if (upload.status == UPLOAD_FILE_START) {
      Serial.printf("Update: %sn", upload.filename.c_str());
      if (!Update.begin(UPDATE_SIZE_UNKNOWN)) { //start with max available size
        Update.printError(Serial);
      }
    } else if (upload.status == UPLOAD_FILE_WRITE) {
      /* flashing firmware to ESP*/
      if (Update.write(upload.buf, upload.currentSize) != upload.currentSize) {
        Update.printError(Serial);
      }
    } else if (upload.status == UPLOAD_FILE_END) {
      if (Update.end(true)) { //true to set the size to the current progress
        Serial.printf("Update Success: %unRebooting...n", upload.totalSize);
      } else {
        Update.printError(Serial);
      }
    }
  });
  server.begin();
}

void loop(void) {
  server.handleClient();
  delay(1);
}

并对其中的代码做如下更改:
给ESP32开发板连接到自己电脑所在的局域网的路由器WIFI上,填上WIFI名称ssis和密码password

const char* host = "esp32";
const char* ssid = "YXDZ1";
const char* password = "YXDZ#2023#1";

host为ESP32作为Web服务器时,将ESP32以易记的主机名(如esp32.local)广播到局域网,其他设备可直接通过该名称访问,用户可通过http://esp32.local直接访问,无需记忆IP地址

const char* host = "esp32";

  /*use mdns for host name resolution*/
  if (!MDNS.begin(host)) { //http://esp32.local
    Serial.println("Error setting up MDNS responder!");
    while (1) {
      delay(1000);
    }
  }

更改好OTA无线局域网网页版入口的用户名和密码,这里都更改为ESP32-OTA

"if(form.userid.value=='ESP32-OTA' && form.pwd.value=='ESP32-OTA')"

通过有线方式连接电脑和ESP32开发板,并把代码编译通过先通过有线方式下载到ESP32开发板,让开发板创建无线OTA远程升级网页版入口
打开串口助手,按下ESP32开发板复位,会显示出所连接的局域网的IP地址,mDNS域名访问创建成功。
在这里插入图片描述
在浏览器输入IP地址或创建的http://esp32.local域名即可跳转到OTA远程升级网页版入口,输入创建好的用户名和密码并登录
在这里插入图片描述
此时再跳转到程序下载界面,选择编译好的程序文件并上传到开发板内
在这里插入图片描述

创建一个ESP32开发板进行OTA升级应用程序示例

若要每次都要对ESP32开发板进行OTA升级,新的应用程序中都要加入上述OTA无线局域网网页版远程升级入口的相关代码。下面为一个ESP32开发板GPIO2引脚上的闪灯程序的代码,并加入了OTA的相关代码,然后对OTA部分的代码进行上述一样的更改。

#include < WiFi.h >
#include < WiFiClient.h >
#include < WebServer.h >
#include < ESPmDNS.h >
#include < Update.h >

const char* host = "esp32";
const char* ssid = "YXDZ1";
const char* password = "YXDZ#2023#1";

//variabls to blink without delay:
const int led = 2;
unsigned long previousMillis = 0;        // will store last time LED was updated
const long interval = 1000;           // interval at which to blink (milliseconds)
int ledState = LOW;             // ledState used to set the LED

WebServer server(80);

/*
 * Login page
 */

const char* loginIndex = 
 "< form name='loginForm' >"
    "< table width='20%' bgcolor='A09F9F' align='center' >"
        "< tr >"
            "< td colspan=2 >"
                "< center >< font size=4 >ESP32 Login Page< /font >< /center >"
                "
"
            "< /td >"
            "
"
            "
"
        "< /tr >"
        "< td >Username:< /td >"
        "< td >< input type='text' size=25 name='userid' >
< /td >"
        "< /tr >"
        "
"
        "
"
        "< tr >"
            "< td >Password:< /td >"
            "< td >< input type='Password' size=25 name='pwd' >
< /td >"
            "
"
            "
"
        "< /tr >"
        "< tr >"
            "< td >< input type='submit' onclick='check(this.form)' value='Login' >< /td >"
        "< /tr >"
    "< /table >"
"< /form >"
"< script >"
    "function check(form)"
    "{"
    "if(form.userid.value=='ESP32-OTA' && form.pwd.value=='ESP32-OTA')"
    "{"
    "window.open('/serverIndex')"
    "}"
    "else"
    "{"
    " alert('Error Password or Username')/*displays error message*/"
    "}"
    "}"
"< /script >";
 
/*
 * Server Index Page
 */
 
const char* serverIndex = 
"< script src='/images/chaijie_default.png' >< /script >"
"< form method='POST' action='#' enctype='multipart/form-data' id='upload_form' >"
   "< input type='file' name='update' >"
        "< input type='submit' value='Update' >"
    "< /form >"
 "progress: 0%"
 "< script >"
  "$('form').submit(function(e){"
  "e.preventDefault();"
  "var form = $('#upload_form')[0];"
  "var data = new FormData(form);"
  " $.ajax({"
  "url: '/update',"
  "type: 'POST',"
  "data: data,"
  "contentType: false,"
  "processData:false,"
  "xhr: function() {"
  "var xhr = new window.XMLHttpRequest();"
  "xhr.upload.addEventListener('progress', function(evt) {"
  "if (evt.lengthComputable) {"
  "var per = evt.loaded / evt.total;"
  "$('#prg').html('progress: ' + Math.round(per*100) + '%');"
  "}"
  "}, false);"
  "return xhr;"
  "},"
  "success:function(d, s) {"
  "console.log('success!')" 
 "},"
 "error: function (a, b, c) {"
 "}"
 "});"
 "});"
 "< /script >";

/*
 * setup function
 */
void setup(void) {
  pinMode(led, OUTPUT);
  
  Serial.begin(115200);

  // Connect to WiFi network
  WiFi.begin(ssid, password);
  Serial.println("");

  // Wait for connection
  while (WiFi.status() != WL_CONNECTED) {
    delay(500);
    Serial.print(".");
  }
  Serial.println("");
  Serial.print("Connected to ");
  Serial.println(ssid);
  Serial.print("IP address: ");
  Serial.println(WiFi.localIP());

  /*use mdns for host name resolution*/
  if (!MDNS.begin(host)) { //http://esp32.local
    Serial.println("Error setting up MDNS responder!");
    while (1) {
      delay(1000);
    }
  }
  Serial.println("mDNS responder started");
  /*return index page which is stored in serverIndex */
  server.on("/", HTTP_GET, []() {
    server.sendHeader("Connection", "close");
    server.send(200, "text/html", loginIndex);
  });
  server.on("/serverIndex", HTTP_GET, []() {
    server.sendHeader("Connection", "close");
    server.send(200, "text/html", serverIndex);
  });
  /*handling uploading firmware file */
  server.on("/update", HTTP_POST, []() {
    server.sendHeader("Connection", "close");
    server.send(200, "text/plain", (Update.hasError()) ? "FAIL" : "OK");
    ESP.restart();
  }, []() {
    HTTPUpload& upload = server.upload();
    if (upload.status == UPLOAD_FILE_START) {
      Serial.printf("Update: %sn", upload.filename.c_str());
      if (!Update.begin(UPDATE_SIZE_UNKNOWN)) { //start with max available size
        Update.printError(Serial);
      }
    } else if (upload.status == UPLOAD_FILE_WRITE) {
      /* flashing firmware to ESP*/
      if (Update.write(upload.buf, upload.currentSize) != upload.currentSize) {
        Update.printError(Serial);
      }
    } else if (upload.status == UPLOAD_FILE_END) {
      if (Update.end(true)) { //true to set the size to the current progress
        Serial.printf("Update Success: %unRebooting...n", upload.totalSize);
      } else {
        Update.printError(Serial);
      }
    }
  });
  server.begin();
}

void loop(void) {
  server.handleClient();
  delay(1);

  //loop to blink without delay
  unsigned long currentMillis = millis();

  if (currentMillis - previousMillis >= interval) {
    // save the last time you blinked the LED
    previousMillis = currentMillis;

    // if the LED is off turn it on and vice-versa:
    ledState = not(ledState);

    // set the LED with the ledState of the variable:
    digitalWrite(led, ledState);
  }
}

ESP32开发板OTA升级下载应用程序

编译并导出可供OTA升级下载的二进制bin应用程序文件
在这里插入图片描述
在项目文件夹下的build文件夹里可看到可供下载的bin应用程序文件
在这里插入图片描述
进入OTA无线局域网网页版远程升级入口,打开应用程序bin文件,并对ESP32开发板进行远程下载程序

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 无线局域网
    +关注

    关注

    1

    文章

    240

    浏览量

    30423
  • 程序
    +关注

    关注

    117

    文章

    3827

    浏览量

    83078
  • OTA
    OTA
    +关注

    关注

    7

    文章

    614

    浏览量

    36652
  • ESP32
    +关注

    关注

    21

    文章

    1022

    浏览量

    19362
收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    GD32单片机STM32远程下载手机程序升级固件下载局域网网页升级工具

    GD32、STM32单片机,是我们最常见的一种MCU。通常我们在使用STM32单片机都会遇到程序在线升级下载的问题。使用该方法可以完成手机网页在线程序
    的头像 发表于 11-09 12:31 ?2836次阅读
    GD32单片机STM32<b class='flag-5'>远程</b><b class='flag-5'>下载</b>手机<b class='flag-5'>程序</b><b class='flag-5'>升级</b>固件<b class='flag-5'>下载</b><b class='flag-5'>局域网</b>网页<b class='flag-5'>升级</b>工具

    浅析思科无线局域网

    思科无线局域网(WLAN)基础。  无线局域网(WLAN)  11.1 WLAN基础  WLAN(Wireless Local Area Network,
    发表于 07-01 07:04

    嵌人式系统的无线局域网接入怎么实现?

    设备驱动的深入理解和分析,成功地移植在Atmel 9261 ARM处理器上。实现了嵌人式系统的无线局域网接入。利用该平台,可以进一步设计完善医用伽马相机和小型SPECT设备的手持数据采集系统,使得控制人员能够远离数据采集现场,而通过远程
    发表于 03-06 06:27

    乐鑫ESP32空中下载(OTA)解决方案实操

    前言 OTA(Over-the-AirTechnology)即空中下载技术,是通过移动通信的空中接口实现对移动终端设备及SIM卡数据进行远程管理的技术。
    发表于 06-30 10:39

    OTA的具体应用场景及远程升级远程的含义具体是什么?

    OTA远程升级一直有一个疑问,希望各位道友解答一下。不胜感激疑问1通过看官方的OTA升级的文档
    发表于 11-14 14:21

    如何用esp32组建局域网

    大家好,我想用esp32组建一个局域网,请大家帮我参考一下,要求如下:一个局域网内数量不超过100台。一台主机,可以同其他的设备通讯,其余均为从机,只能和主机通讯,从机之间不能直接通讯。通讯的有效
    发表于 03-09 08:02

    无线局域网(WLAN)是什么?

    据传输。   WLAN 的工作原理   无线局域网的工作原理是通过发送和接收无线电信号来完成数据传输。接入点将有线网络连接到无线网络,允许
    发表于 05-17 17:11

    GD32单片机STM32远程下载手机程序升级固件下载局域网网页升级工具

    ,即可浏览到上传的文件。点击文件后面对应的Flash按钮即可完成STM32单片机的在线升级。 该服务器还支持FTP方式远程下载固件的功能,能够完成局域网、互联网的
    发表于 11-10 15:03

    无线局域网简介

    无线局域网无线局域网络(Wireless Local Area Networks; WLAN)是相当便利的数据传输系统,它利用射频(Radio Frequency; RF)的技术,
    发表于 11-24 03:21 ?2448次阅读

    浅谈无线局域网的优点

    无线局域网,其英文缩写为WLAN。无线局域网无线通信技术与网络技术相结合的产物。从专业角度讲,无线
    发表于 11-14 15:51 ?3326次阅读

    解答无线局域网该如何设置

    近些年来,网络技术发展日新月异,随着无线技术的不断发展,现在我们生活当中的网络,以及从过去的局域网发展成现在的无线局域网无线
    发表于 11-14 19:17 ?4045次阅读

    无线局域网的优点有哪些

    无线局域网是计算机网络与无线通信技术相结合的产物,无线局域网的基础还是传统的有线局域网,是有线
    的头像 发表于 01-10 08:56 ?9888次阅读

    ESP32通信amp;局域网刺破

    ESP32局域网中通讯非常简单,按照模块的AT指令集发送指令即可。常规情况下,需要局域网内部的IP与局域网外界通讯,需要穿透局域网,此时需
    的头像 发表于 02-13 13:38 ?2098次阅读

    wlan是无线网还是局域网 wlan包括哪些无线局域网协议

    WLAN是无线局域网(Wireless Local Area Network)的缩写,意味着它是一种在有限范围内使用无线技术进行数据传输的局域网。相比有线
    发表于 06-14 17:18 ?3590次阅读

    ESP32ESP32通过Internet进行通信

    电子发烧友网站提供《ESP32ESP32通过Internet进行通信.zip》资料免费下载
    发表于 06-15 09:58 ?5次下载
    <b class='flag-5'>ESP32</b>到<b class='flag-5'>ESP32</b><b class='flag-5'>通过</b>Internet进行通信