草庐IT

ESP8266--Arduino开发(搭建HTTP网络服务器)

贝勒里恩 2024-05-03 原文

文章目录

一、前言

超文本传输协议(Hyper Text Transfer Protocol,HTTP)是一个简单的请求-响应协议,它通常运行在TCP之上,它指定了客户端可能发送给服务器什么样的消息以及得到什么样的响应,,我们使用浏览器打开的网页使用的就是HTTP协议。

接下来我们会参数在ESP8266-NodeMCU上建立一个HTTP网络服务器,然后通过浏览器来访问它。


二、搭建HTTP网络服务器

/*
  ESP8266-NodeMCU作为HttpServer服务器
*/
 
#include <ESP8266WiFi.h>                        // 本程序使用ESP8266WiFi库
#include <ESP8266WebServer.h>                   // web服务器通信库需要使用

/* 1. 设置Wifi接入信息 */
const char* ssid     = "LaiFu";                // 需要连接到的WiFi名
const char* password = "wangjichuan";          // 连接的WiFi密码

/* 2. 创建一个web服务器对象,使用80端口,HTTP网络服务器标准端口号即为80 */
ESP8266WebServer esp8266_server(80);

/* 3. 处理访问网站根目录“/”的访问请求 */
void handleRoot() { 
  esp8266_server.send(200, "text/plain", "Hello from ESP8266");     // NodeMCU将调用此函数。
}

/* 4. 设置处理404情况的函数'handleNotFound' */
void handleNotFound(){                                              // 当浏览器请求的网络资源无法在服务器找到时,
  esp8266_server.send(404, "text/plain", "404: Not found");         // NodeMCU将调用此函数。
}

void setup() {
  /* 1. 初始化串口通讯波特率为115200*/
  Serial.begin(115200);

  /* 2. 开启wifi连接,连接成功后打印IP地址 */
  WiFi.mode(WIFI_STA);                          // 设置Wifi工作模式为STA,默认为AP+STA模式
  WiFi.begin(ssid, password);                   // 通过wifi名和密码连接到Wifi
  Serial.print("\r\nConnecting to ");           // 串口监视器输出网络连接信息
  Serial.print(ssid); Serial.println(" ...");   // 显示NodeMCU正在尝试WiFi连接
  int i = 0;                                    // 检查WiFi是否连接成功
  while (WiFi.status() != WL_CONNECTED)         // WiFi.status()函数的返回值是由NodeMCU的WiFi连接状态所决定的。 
  {                                             // 如果WiFi连接成功则返回值为WL_CONNECTED
    delay(1000);                                // 此处通过While循环让NodeMCU每隔一秒钟检查一次WiFi.status()函数返回值
    Serial.print("waiting for ");                          
    Serial.print(i++); Serial.println("s...");       
  }                                             
  Serial.println("");                           // WiFi连接成功后
  Serial.println("WiFi connected!");            // NodeMCU将通过串口监视器输出"连接成功"信息。
  Serial.print("IP address: ");                 // 同时还将输出NodeMCU的IP地址。这一功能是通过调用
  Serial.println(WiFi.localIP());               // WiFi.localIP()函数来实现的。该函数的返回值即NodeMCU的IP地址。

  /* 3. 开启http网络服务器功能 */
  esp8266_server.begin();                       // 启动http网络服务器
  esp8266_server.on("/", handleRoot);           // 设置请求根目录时的处理函数函数
  esp8266_server.onNotFound(handleNotFound);    // 设置无法响应时的处理函数    
  Serial.println("HTTP esp8266_server started");// 告知用户ESP8266网络服务功能已经启动
}

void loop() {
  esp8266_server.handleClient();                // 处理http访问,需要一直运行
}

ESP8266连接手机热点

打开手机浏览器,输入ESP8266的IP

显示Hello FROM esp8266字符串说明HTTP服务器访问正常


三、添加一个简单网页进行交互

通过上面的步骤只是搭建好了HTTP网络服务器的框架,现在嵌入一个简单的网页来试试。

这个网页提供两个按钮:关灯、开灯

主要是要添加路径响应函数,在函数中设置小灯的亮灭

esp8266_server.on()中的路径对应于表单的提交按钮action属性的路径

/*
  ESP8266-NodeMCU作为HttpServer服务器
*/
 
#include <ESP8266WiFi.h>                        // 本程序使用ESP8266WiFi库
#include <ESP8266WebServer.h>                   // web服务器通信库需要使用

/* 1. 设置Wifi接入信息 */
const char* ssid     = "LaiFu";                // 需要连接到的WiFi名
const char* password = "wangjichuan";          // 连接的WiFi密码

/* 2. 创建一个web服务器对象,使用80端口,HTTP网络服务器标准端口号即为80 */
ESP8266WebServer esp8266_server(80);

/* 3. 处理访问网站根目录“/”的访问请求 */
void handleRoot() { 
  String htmlCode = "<!DOCTYPE html>\n"; 
  htmlCode +=       " <html>\n";
  htmlCode +=       "   <head>\n";
  htmlCode +=       "     <meta charset=\"UTF-8\"/>\n";
  htmlCode +=       "     <title>ESP8266 Butoon Ctrl</title>\n";
  htmlCode +=       "   </head>\n"; 
  htmlCode +=       "   <body>\n";
  htmlCode +=       "     <h2 align=\"center\">esp8266控制开关</h2>";
  htmlCode +=       "     <p><form action=\"/LED_OFF\" method=\"POST\" align=\"center\"><input type=\"submit\" value=\"关灯\"></form></p>\n";
  htmlCode +=       "     <p><form action=\"/LED_ON\" method=\"POST\" align=\"center\"><input type=\"submit\" value=\"开灯\"></form></p>\n";
  htmlCode +=       "   </body>\n";
   htmlCode +=      " </html>\n";
  esp8266_server.send(200, "text/html", htmlCode);     // NodeMCU将调用此函数。
}

/* 4. 设置处理404情况的函数'handleNotFound' */
void handleNotFound(){                                              // 当浏览器请求的网络资源无法在服务器找到时,
  esp8266_server.send(404, "text/plain", "404: Not found");         // NodeMCU将调用此函数。
}

void handle_LED_ON() {
  Serial.println("handle_LED_ON");
  digitalWrite(2, LOW);
}

void handle_LED_OFF() {
  Serial.println("handle_LED_OFF");
  digitalWrite(2, HIGH);
}

void setup() {
  /* 1. 初始化串口通讯波特率为115200*/
  Serial.begin(115200);

  //配置GPIO2为输出模式
  pinMode(2, OUTPUT);
  digitalWrite(2, LOW);

  /* 2. 开启wifi连接,连接成功后打印IP地址 */
  WiFi.mode(WIFI_STA);                          // 设置Wifi工作模式为STA,默认为AP+STA模式
  WiFi.begin(ssid, password);                   // 通过wifi名和密码连接到Wifi
  Serial.print("\r\nConnecting to ");           // 串口监视器输出网络连接信息
  Serial.print(ssid); Serial.println(" ...");   // 显示NodeMCU正在尝试WiFi连接
  int i = 0;                                    // 检查WiFi是否连接成功
  while (WiFi.status() != WL_CONNECTED)         // WiFi.status()函数的返回值是由NodeMCU的WiFi连接状态所决定的。 
  {                                             // 如果WiFi连接成功则返回值为WL_CONNECTED
    delay(1000);                                // 此处通过While循环让NodeMCU每隔一秒钟检查一次WiFi.status()函数返回值
    Serial.print("waiting for ");                          
    Serial.print(i++); Serial.println("s...");       
  }                                             
  Serial.println("");                           // WiFi连接成功后
  Serial.println("WiFi connected!");            // NodeMCU将通过串口监视器输出"连接成功"信息。
  Serial.print("IP address: ");                 // 同时还将输出NodeMCU的IP地址。这一功能是通过调用
  Serial.println(WiFi.localIP());               // WiFi.localIP()函数来实现的。该函数的返回值即NodeMCU的IP地址。

  /* 3. 开启http网络服务器功能 */
  esp8266_server.begin();                       // 启动http网络服务器
  esp8266_server.on("/", handleRoot);           // 设置请求根目录时的处理函数函数
  esp8266_server.onNotFound(handleNotFound);    // 设置无法响应时的处理函数    

  esp8266_server.on("/LED_ON", handle_LED_ON);  // 设置请求开灯目录时的处理函数函数
  esp8266_server.on("/LED_OFF", handle_LED_OFF);// 设置请求关灯目录时的处理函数函数
  
  Serial.println("HTTP esp8266_server started");// 告知用户ESP8266网络服务功能已经启动
}

void loop() {
  esp8266_server.handleClient();                // 处理http访问,需要一直运行
}

在处理根目录访问请求函数handleRoot()中添加了html字符串,所以打开ESP8266的IP就能看到解析的网页:

  • 点击开灯可以看见开发板小灯打开;
  • 点击关灯可以看见开发板小灯关闭;


四、ESP8266WebServer库

4.1、WebServer管理方法

1、创建Web Server

ESP8266WebServer(IPAddress addr, int port = 80);
ESP8266WebServer(int port = 80);

//例如:创建一个web服务器对象,使用80端口
ESP8266WebServer esp8266_server(80);
  • addr:IP地址
  • port:端口号

2、启动Web Server

void begin();
void begin(uint16_t port);

//例如:启动Web Server
esp8266_server.begin(); 
  • port:端口号
  • begin函数要在配置好各个请求后使用

3、关闭Web Server

close();
stop();

//例如:关闭Web Server
esp8266_server.close(); 

4.2、配置client请求处理方法

1、请求响应回调

void on(const String &url, THandlerFunction handler);
  • url:路径
  • handler:对应url的处理函数

注意:这里的handler函数是Http_ANY,不区分GET、POST等

void on(const String &url, HTTPMethod method, THandlerFunction fn);
  • url:路径
  • method:Http请求方法(HTTP_ANY, HTTP_GET, HTTP_POST, HTTP_PUT, HTTP_PATCH, TTP_DELETE, HTTP_OPTIONS)
  • fn:对应url的处理函数
void on(const String &url, HTTPMethod method, THandlerFunction fn ThandlerFunction ufn);
  • url:路径
  • method:Http请求方法(HTTP_ANY, HTTP_GET, HTTP_POST, HTTP_PUT, HTTP_PATCH, TTP_DELETE, HTTP_OPTIONS)
  • fn:对应url的处理函数
  • ufn:文件上传处理函数

2、配置无效url的handler

void onNotFound(THandlerFunction fn);
  • fn:对应的处理函数

注意:当找不到相对于的http请求处理函数时会调用该函数配置的fn方法

3、配置处理文件上传的handler

void onFileUpload(THandlerFunction fn);
  • fn:对应的处理函数

4.3、获取请求方法

1、获取请求的url

String url();

2、获取请求方法

HTTPMethod method()
  • 返回值: HTTP_ANY, HTTP_GET, HTTP_POST, HTTP_PUT, HTTP_PATCH, HTTP_DELETE, HTTP_OPTIONS

3、获取请求参数的值

String arg(String name);
  • name:根据关键字name获取请求参数的值
String arg(int i);
  • i:获取第i个请求参数的值

4、获取请求参数的名称

String arg(int i);
  • i:获取第i个请求参数的名称

5、获取参数个数

int args();

6、是否存在某个参数

bool hasArg(String name);
  • name:参数的名称

7、设置需要收集的请求头

void collectHeaders(const char* headerKeys[], const size_t headerKeysCount); 
  • headerkeys[]:请求头的名字
  • headerkeysCount:请求头的个数

8、获取请求头参数

String header(String name);
  • name:请求头名称
String header(int i);
  • i:获取第i个请求头参数

9、获取请求头名字

String headerName(int i);
  • i:获取第i个请求头名字

10、获取请求头个数

int headers();

11、判断是否存在某个请求头

bool hasHeader(String name);

12、获取请求头Host的值

String hostHeader();

13、认证校验

bool authenticate(const char * username, const char * password);
  • username: 用户账号
  • password: 用户密码

14、处理http请求

void handleClient();

4.4、获取client请求方法

1、处理文件上传

HTTPUpload& upload();
//实例说明 非完整代码,无法直接运行,理解即可
/**
 * 处理文件上传 HandlerFunction
 * 此方法会在文件上传过程中多次回调,我们可以判断上传状态
 */
void handleFileUpload() {
  //判断http requestUri
  if (server.uri() != "/edit") {
    return;
  }
  //获得 Http上传文件处理对象
  HTTPUpload& upload = server.upload();
  //文件开始上传
  if (upload.status == UPLOAD_FILE_START) {
    String filename = upload.filename;
    if (!filename.startsWith("/")) {
      filename = "/" + filename;
    }
    DBG_OUTPUT_PORT.print("handleFileUpload Name: "); DBG_OUTPUT_PORT.println(filename);
    //本地文件系统创建一个文件用来保存内容
    fsUploadFile = SPIFFS.open(filename, "w");
    filename = String();
  } else if (upload.status == UPLOAD_FILE_WRITE) {
    //文件开始写入文件
    //DBG_OUTPUT_PORT.print("handleFileUpload Data: "); DBG_OUTPUT_PORT.println(upload.currentSize);
    if (fsUploadFile) {
      //写入文件
      fsUploadFile.write(upload.buf, upload.currentSize);
    }
  } else if (upload.status == UPLOAD_FILE_END) {
    //文件上传结束
    if (fsUploadFile) {
      fsUploadFile.close();
    }
    DBG_OUTPUT_PORT.print("handleFileUpload Size: "); DBG_OUTPUT_PORT.println(upload.totalSize);
  }
}
 
//注册文件上传处理回调
server.on("/edit", HTTP_POST, []() {
    server.send(200, "text/plain", "");
  }, handleFileUpload);

2、设置响应头

void sendHeader(const String& name, const String& value, bool first = false);
  • name: 响应头名
  • value: 响应头值
  • first: 是否需要放在第一行

3、设置响应体长度

void setContentLength(const size_t contentLength);

4、发送响应内容

void sendContent(const String& content);
void sendContent_P(PGM_P content);
void sendContent_P(PGM_P content, size_t size);

5、发送响应文件流

size_t streamFile(T &file, const String& contentType);
  • file: 具体文件
  • contentType: 响应类型

6、发送响应数据

void send(int code, const char* content_type = NULL, const String& content = String(""));
void send(int code, char* content_type, const String& content);
void send(int code, const String& content_type, const String& content);
void send_P(int code, PGM_P content_type, PGM_P content);
void send_P(int code, PGM_P content_type, PGM_P content, size_t contentLength);
  • code: 响应状态码
  • content_type: 响应内容类型
  • content:具体响应内容

有关ESP8266--Arduino开发(搭建HTTP网络服务器)的更多相关文章

  1. ruby - 使用 ruby​​ 和 savon 的 SOAP 服务 - 2

    我正在尝试使用ruby​​和Savon来使用网络服务。测试服务为http://www.webservicex.net/WS/WSDetails.aspx?WSID=9&CATID=2require'rubygems'require'savon'client=Savon::Client.new"http://www.webservicex.net/stockquote.asmx?WSDL"client.get_quotedo|soap|soap.body={:symbol=>"AAPL"}end返回SOAP异常。检查soap信封,在我看来soap请求没有正确的命名空间。任何人都可以建议我

  2. ruby - 具有身份验证的私有(private) Ruby Gem 服务器 - 2

    我想安装一个带有一些身份验证的私有(private)Rubygem服务器。我希望能够使用公共(public)Ubuntu服务器托管内部gem。我读到了http://docs.rubygems.org/read/chapter/18.但是那个没有身份验证-如我所见。然后我读到了https://github.com/cwninja/geminabox.但是当我使用基本身份验证(他们在他们的Wiki中有)时,它会提示从我的服务器获取源。所以。如何制作带有身份验证的私有(private)Rubygem服务器?这是不可能的吗?谢谢。编辑:Geminabox问题。我尝试“捆绑”以安装新的gem..

  3. ruby - 使用 C 扩展开发 ruby​​gem 时,如何使用 Rspec 在本地进行测试? - 2

    我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当

  4. ruby - 如何模拟 Net::HTTP::Post? - 2

    是的,我知道最好使用webmock,但我想知道如何在RSpec中模拟此方法:defmethod_to_testurl=URI.parseurireq=Net::HTTP::Post.newurl.pathres=Net::HTTP.start(url.host,url.port)do|http|http.requestreq,foo:1endresend这是RSpec:let(:uri){'http://example.com'}specify'HTTPcall'dohttp=mock:httpNet::HTTP.stub!(:start).and_yieldhttphttp.shou

  5. ruby-on-rails - 启动 Rails 服务器时 ImageMagick 的警告 - 2

    最近,当我启动我的Rails服务器时,我收到了一长串警告。虽然它不影响我的应用程序,但我想知道如何解决这些警告。我的估计是imagemagick以某种方式被调用了两次?当我在警告前后检查我的git日志时。我想知道如何解决这个问题。-bcrypt-ruby(3.1.2)-better_errors(1.0.1)+bcrypt(3.1.7)+bcrypt-ruby(3.1.5)-bcrypt(>=3.1.3)+better_errors(1.1.0)bcrypt和imagemagick有关系吗?/Users/rbchris/.rbenv/versions/2.0.0-p247/lib/ru

  6. ruby-on-rails - s3_direct_upload 在生产服务器中不工作 - 2

    在Rails4.0.2中,我使用s3_direct_upload和aws-sdkgems直接为s3存储桶上传文件。在开发环境中它工作正常,但在生产环境中它会抛出如下错误,ActionView::Template::Error(noimplicitconversionofnilintoString)在View中,create_cv_url,:id=>"s3_uploader",:key=>"cv_uploads/{unique_id}/${filename}",:key_starts_with=>"cv_uploads/",:callback_param=>"cv[direct_uplo

  7. Ruby Sinatra 配置用于生产和开发 - 2

    我已经在Sinatra上创建了应用程序,它代表了一个简单的API。我想在生产和开发上进行部署。我想在部署时选择,是开发还是生产,一些方法的逻辑应该改变,这取决于部署类型。是否有任何想法,如何完成以及解决此问题的一些示例。例子:我有代码get'/api/test'doreturn"Itisdev"end但是在部署到生产环境之后我想在运行/api/test之后看到ItisPROD如何实现? 最佳答案 根据SinatraDocumentation:EnvironmentscanbesetthroughtheRACK_ENVenvironm

  8. ruby - 是否可以覆盖 gemfile 进行本地开发? - 2

    我们的git存储库中目前有一个Gemfile。但是,有一个gem我只在我的环境中本地使用(我的团队不使用它)。为了使用它,我必须将它添加到我们的Gemfile中,但每次我checkout到我们的master/dev主分支时,由于与跟踪的gemfile冲突,我必须删除它。我想要的是类似Gemfile.local的东西,它将继承从Gemfile导入的gems,但也允许在那里导入新的gems以供使用只有我的机器。此文件将在.gitignore中被忽略。这可能吗? 最佳答案 设置BUNDLE_GEMFILE环境变量:BUNDLE_GEMFI

  9. ruby - 在 Windows 机器上使用 Ruby 进行开发是否会适得其反? - 2

    这似乎非常适得其反,因为太多的gem会在window上破裂。我一直在处理很多mysql和ruby​​-mysqlgem问题(gem本身发生段错误,一个名为UnixSocket的类显然在Windows机器上不能正常工作,等等)。我只是在浪费时间吗?我应该转向不同的脚本语言吗? 最佳答案 我在Windows上使用Ruby的经验很少,但是当我开始使用Ruby时,我是在Windows上,我的总体印象是它不是Windows原生系统。因此,在主要使用Windows多年之后,开始使用Ruby促使我切换回原来的系统Unix,这次是Linux。Rub

  10. ruby - 用 Ruby 编写一个简单的网络服务器 - 2

    我想在Ruby中创建一个用于开发目的的极其简单的Web服务器(不,不想使用现成的解决方案)。代码如下:#!/usr/bin/rubyrequire'socket'server=TCPServer.new('127.0.0.1',8080)whileconnection=server.acceptheaders=[]length=0whileline=connection.getsheaders想法是从命令行运行这个脚本,提供另一个脚本,它将在其标准输入上获取请求,并在其标准输出上返回完整的响应。到目前为止一切顺利,但事实证明这真的很脆弱,因为它在第二个请求上中断并出现错误:/usr/b

随机推荐