2个小时0基础,手搓一个智能wifi温湿度计

复制代码
ESP8266 + SHT30
↓
连接 WiFi
↓
浏览器查看温湿度
↓
网页 OTA 更新固件
↓
后续再加信息推送


后续添加对接天猫精灵的功能和空调联动功能

南方天气阴冷潮湿,墙体发霉

上代码

复制代码
#include <ESP8266WiFi.h>
#include <ESP8266WebServer.h>
#include <ESP8266HTTPUpdateServer.h>
#include <ESP8266mDNS.h>
#include <Wire.h>

#define SHT30_ADDR 0x44

const char* ssid = "home-wifi";
const char* password = "123456";

// 网页 OTA 登录账号密码
const char* otaUser = "admin";
const char* otaPass = "admin123456";

ESP8266WebServer server(80);
ESP8266HTTPUpdateServer httpUpdater;

float temperature = 0.0;
float humidity = 0.0;
bool sht30Ok = false;

uint8_t crc8(const uint8_t *data, int len) {
  uint8_t crc = 0xFF;
  for (int i = 0; i < len; i++) {
    crc ^= data[i];
    for (int j = 0; j < 8; j++) {
      if (crc & 0x80) {
        crc = (crc << 1) ^ 0x31;
      } else {
        crc <<= 1;
      }
    }
  }
  return crc;
}

bool readSHT30() {
  Wire.beginTransmission(SHT30_ADDR);
  Wire.write(0x2C);
  Wire.write(0x06);
  uint8_t error = Wire.endTransmission();

  if (error != 0) {
    return false;
  }

  delay(20);

  Wire.requestFrom(SHT30_ADDR, 6);

  if (Wire.available() != 6) {
    return false;
  }

  uint8_t data[6];
  for (int i = 0; i < 6; i++) {
    data[i] = Wire.read();
  }

  if (crc8(data, 2) != data[2] || crc8(data + 3, 2) != data[5]) {
    return false;
  }

  uint16_t rawTemp = (data[0] << 8) | data[1];
  uint16_t rawHum  = (data[3] << 8) | data[4];

  temperature = -45.0 + 175.0 * rawTemp / 65535.0;
  humidity = 100.0 * rawHum / 65535.0;

  return true;
}

void handleRoot() {
  readSHT30();

  String html = "";
  html += "<!DOCTYPE html><html><head>";
  html += "<meta charset='UTF-8'>";
  html += "<meta name='viewport' content='width=device-width, initial-scale=1'>";
  html += "<meta http-equiv='refresh' content='5'>";
  html += "<title>ESP8266 SHT30</title>";
  html += "<style>";
  html += "body{font-family:Arial;background:#f4f4f4;text-align:center;padding:30px;}";
  html += ".card{background:white;padding:25px;border-radius:16px;display:inline-block;box-shadow:0 4px 16px #ccc;min-width:260px;}";
  html += ".value{font-size:36px;font-weight:bold;margin:15px;}";
  html += "a{display:block;margin-top:20px;color:#0366d6;text-decoration:none;}";
  html += "</style>";
  html += "</head><body>";
  html += "<div class='card'>";
  html += "<h2>ESP8266 SHT30</h2>";

  if (sht30Ok) {
    html += "<div>温度</div>";
    html += "<div class='value'>" + String(temperature, 2) + " ℃</div>";
    html += "<div>湿度</div>";
    html += "<div class='value'>" + String(humidity, 2) + " %RH</div>";
  } else {
    html += "<h3>SHT30 读取失败</h3>";
  }

  html += "<p>IP: " + WiFi.localIP().toString() + "</p>";
  html += "<p>每 5 秒自动刷新</p>";
  html += "<a href='/json'>查看 JSON 数据</a>";
  html += "<a href='/update'>网页 OTA 更新</a>";
  html += "</div>";
  html += "</body></html>";

  server.send(200, "text/html; charset=utf-8", html);
}

void handleJson() {
  readSHT30();

  String json = "{";
  json += "\"ok\":";
  json += sht30Ok ? "true" : "false";
  json += ",";
  json += "\"temperature\":";
  json += String(temperature, 2);
  json += ",";
  json += "\"humidity\":";
  json += String(humidity, 2);
  json += ",";
  json += "\"ip\":\"";
  json += WiFi.localIP().toString();
  json += "\"";
  json += "}";

  server.send(200, "application/json; charset=utf-8", json);
}

void setup() {
  Serial.begin(115200);
  delay(1000);

  Wire.begin(4, 5); // SDA=D2/GPIO4, SCL=D1/GPIO5

  Serial.println();
  Serial.println("Connecting WiFi...");

  WiFi.mode(WIFI_STA);
  WiFi.begin(ssid, password);

  int retry = 0;
  while (WiFi.status() != WL_CONNECTED && retry < 60) {
    delay(500);
    Serial.print(".");
    retry++;
  }

  Serial.println();

  if (WiFi.status() == WL_CONNECTED) {
    Serial.println("WiFi connected");
    Serial.print("IP address: ");
    Serial.println(WiFi.localIP());
  } else {
    Serial.println("WiFi connect failed");
  }

  if (MDNS.begin("sht30")) {
    Serial.println("mDNS started: http://sht30.local");
  }

  server.on("/", handleRoot);
  server.on("/json", handleJson);

  // 网页 OTA 地址:http://设备IP/update
  httpUpdater.setup(&server, "/update", otaUser, otaPass);

  server.begin();
  Serial.println("HTTP server started");

  Serial.println("Open:");
  Serial.print("http://");
  Serial.println(WiFi.localIP());
  Serial.println("or:");
  Serial.println("http://sht30.local");
  Serial.println("OTA:");
  Serial.print("http://");
  Serial.print(WiFi.localIP());
  Serial.println("/update");
}

void loop() {
  sht30Ok = readSHT30();

  server.handleClient();
  MDNS.update();

  static unsigned long lastPrint = 0;
  if (millis() - lastPrint > 3000) {
    lastPrint = millis();

    if (sht30Ok) {
      Serial.print("Temperature: ");
      Serial.print(temperature, 2);
      Serial.println(" °C");

      Serial.print("Humidity: ");
      Serial.print(humidity, 2);
      Serial.println(" %RH");

      Serial.print("IP: ");
      Serial.println(WiFi.localIP());

      Serial.println("------------------");
    } else {
      Serial.println("SHT30 read failed");
    }
  }
}

ota升级

www.jxzhiqu.cn

相关推荐
数字新视界1 小时前
DCIM管理系统是什么?它的智能化能力与应用场景有哪些?
物联网·数据中心·dcim·动环监控·新人首发
时光の尘1 小时前
【嵌入式大厂面经】·IIC常见考点(持续更新中···)
arm开发·单片机·嵌入式硬件·mcu·物联网·iot
高翔·权衡之境2 小时前
主题3:天线与耦合——近场与远场
网络·嵌入式硬件·物联网·软件工程·信息与通信
MicroTech20252 小时前
微算法科技(NASDAQ: MLGO)区块链赋能物联网:量子安全互认证创新方案
科技·物联网·区块链
武汉知识图谱科技3 小时前
从“事后记录”到“事前预警”:智能视觉+物联网如何让危险提前3秒发现
物联网
会周易的程序员19 小时前
aiDgeScanner 工业设备网络扫描与管理工具
网络·c++·物联网·架构·electron·node.js·iot
会周易的程序员1 天前
aiDgeScanner架构与实现
c++·ide·物联网·架构·node.js·aiot
Wireless_wifi61 天前
QCN9274-Based 2.4G/5G/6G Wireless Modules Tri-Band WiFi 7
人工智能·物联网·5g·ubuntu
TDengine (老段)1 天前
TDengine 整体架构全景 — 深度解析
大数据·数据库·物联网·架构·时序数据库·tdengine·涛思数据