1. 通过土壤湿度传感器收集湿度数据;
2. 根据湿度数据决定是否浇水;
3. 通过OLED显示相关信息;
4. 可手动进行浇水(不考虑土壤湿度);
1.需要使用多任务来实现,本程序使用ESP32 FreeRTOS
2.在主循环内读取土壤湿度数据并放入队列(任务0)
3.在数据接收任务内接收队列数据,并判断是否要浇水(任务1)
4.使用一个任务来处理按键(任务2)
5.使用一个任务来处理OLED显示(任务3)
/**********************************************************************
** 小鹏浇花系统
*
** version: 1.0.0
** author: 老张
** homesite: www.xpstem.com/product/intelligent-watering/index
**********************************************************************/
#include <Arduino.h>
#include <SSD1306Wire.h> // 4.6.1 (ThingPulse)
#include <OneButton.h> // 2.6.1
#include <ArrayList.h> // 1.0.2
//************** 常数数据 开始 **********************
// Pins
int soil_moisture_sensor_pin = 27; // 土壤湿度传感器引脚
int pump_control_relay_pin = 15; // 水泵继电器引脚
int oled_display_pin_scl = 22; // OLED显示屏I2C引脚
int oled_display_pin_sda = 21; // OLED显示屏I2C引脚
int oled_display_i2c_addr = 61; // OLED显示屏I2C地址
int manual_button_pin = 26; // 手动按钮引脚
//************* 配置变量 结束 *********************
//************* 运行时变量 开始 ********************
OneButton manualBtn(manual_button_pin, true);
int stateCode = 0; //1:config, 2:work, 3:sleep, 9:error
String stateInfo = "";
int collectCount = 0;
int sensorVal = 0;
int Threshold = 50; // 湿度阀值
ArrayList<int> valueList; // 湿度值列表
QueueHandle_t msgQueue = xQueueCreate(10, sizeof(int)); // 消息队列
//************* 运行时变量 结束 *********************
/***********************************************************
*** 控制水泵继电器
**********************************************************/
void main_watering(int relayPin, int seconds) {
pinMode(relayPin, OUTPUT);
digitalWrite(relayPin, HIGH);
Serial.printf("打开水泵继电器,引脚:%d,保持 %d 秒。\n", relayPin, seconds);
for (int i=0; i<seconds; i++) {
Serial.print(".");
delay(1000);
}
Serial.println("");
digitalWrite(relayPin, LOW);
Serial.println("关闭水泵继电器。");
}
/***********************************************************
*** UI任务
**********************************************************/
void task_display(void *pvParam) {
SSD1306Wire display(/* address */ oled_display_i2c_addr,
/* sda */ oled_display_pin_sda,
/* scl */ oled_display_pin_scl);
display.init();
//display.flipScreenVertically();
display.setFont(ArialMT_Plain_16);
while(1) {
display.clear();
if (stateCode==1) {
display.drawString(0, 0, "[config]");
} else if (stateCode==2) {
display.drawString(0, 0, "[work]" );
display.drawString(0, 20, "value: " + (sensorVal==0 ? "" : String(sensorVal)) );
if (stateInfo.length()>0) {
display.drawString(0, 40, stateInfo );
}
} else if (stateCode==3) {
display.drawString(0, 0, "Zzz..." );
} else if (stateCode==9) {
display.drawString(0, 0, "[error]");
display.drawString(0, 20, stateInfo );
}
display.display();
vTaskDelay(1000 / portTICK_PERIOD_MS);
};
}
/********************************************************
*** Button任务
********************************************************/
void task_buttonTick(void *pvParam) {
while(1) {
manualBtn.tick();
vTaskDelay(10 / portTICK_PERIOD_MS); //10ms
}
}
/********************************************************
*** 数据接收任务
********************************************************/
void task_dataReceive(void *pvParam) {
while (1) {
int received;
// 从队列接收数据
if (xQueueReceive(msgQueue, &received, 100/portTICK_PERIOD_MS) == pdTRUE) {
valueList.add(received);
int flag = 0;
if (valueList.size() > 5) {
// 检查是否全部低于阀值
for (int i=0; i<valueList.size(); i++) {
if (valueList.get(i) > Threshold) { // 超过阀值
flag = 1;
break;
}
}
if (flag == 0) {
// 全部低于阀值,则浇水一次
main_watering(pump_control_relay_pin, 5);
}
valueList.remove(0);
}
}
}
}
/***********************************************************
*** 进入休眠状态
**********************************************************/
void main_deepSleep(int sleepSeconds) {
stateCode = 3;
delay(3000); //3s for ui update
//digitalWrite(power_led_pin, LOW);
Serial.printf("进入深度休眠状态(%d%s)。\n",
sleepSeconds<3600 ? sleepSeconds/60 : sleepSeconds/3600,
sleepSeconds<3600 ? "分钟" : "小时");
ESP.deepSleep(sleepSeconds * 1e6);
}
/***********************************************************
*** 收集传感器数据
**********************************************************/
void main_collectData(int sensorPin) {
Serial.printf("获取土壤湿度数据,引脚:%d\n", sensorPin);
int originVal = analogRead(sensorPin);
if (originVal > 0) {
sensorVal = map(originVal, 1, 4095, 100, 1);
// 加入队列
xQueueSend(msgQueue, &sensorVal, portMAX_DELAY);
} else {
// 数据无效,不做处理。
}
}
/***********************************************************
*** 初始化设置
**********************************************************/
void setup() {
Serial.begin(115200);
Serial.println();
Serial.println("-------------------------------");
Serial.println("--------- 小鹏浇花系统 ----------");
Serial.println("-------------------------------");
Serial.println("启动UI任务程序。");
// UI任务
xTaskCreate(
task_display, // 任务函数
"Display_Task", // 任务名称
4096, // 堆栈大小(字节)
NULL, // 参数
1, // 优先级
NULL // 任务句柄
);
Serial.println("启动主程序。");
stateCode = 2;
// 等传感器稳定。
Serial.print("等待传感器初始化(3sec).");
for (int i=0;i<6;i++) {
Serial.print(".");
delay(500);
}
Serial.println("");
// 手动按钮
manualBtn.attachDoubleClick([]() {
main_watering(pump_control_relay_pin, 5);
});
xTaskCreate(task_buttonTick, "ButtonTick_Task", 2048, NULL, 1, NULL);
xTaskCreate(task_dataReceive, "DataReceive_Task", 2048, NULL, 1, NULL);
}
/***********************************************************
*** 主循环
*** 1.获取传感器数据并发送
**********************************************************/
void loop() {
// 采集数据
main_collectData(soil_moisture_sensor_pin);
collectCount++;
vTaskDelay(120000 / portTICK_PERIOD_MS); //120s
if (collectCount > 30) {
main_deepSleep(23 * 3600); //23小时
}
}
二进制(binary),发现者莱布尼茨,是在数学和数字电路中以2为基数的记数系统,是以2为基数代表系统的二进位制。这一系统中,通常用两个不同的符号0(代表零)和1(代表一)来表示 。
ArrayList 类是一个 C++ 模板类,它提供了 ArrayList 的实现,以便轻松存储任何指定类型的值。它允许使用索引进行高效存储和检索,支持排序操作。
本文介绍ESP32中的中断机制,以及如何通过GPIO中断实现按钮控制。重点讲解了如何设置中断服务例程、处理中断抖动问题,并提供了消除中断抖动的示例代码。
本文主要介绍在未联网(AP热点)情况下实现WEB交互界面的CSS和javascript库。
本文介绍如何使用Arduino-ESP32库中的API函数获取ESP32的芯片、RAM信息等,并提供了一个示例程序代码。
ESP32系列(包括ESP32-S3)搭载Xtensa双核处理器,默认情况下Arduino框架仅使用单核运行用户代码,通过多核编程,可以充分利用硬件资源来提升系统响应和性能。
ESP32 芯片有34个物理GPIO管脚。每个GPIO管脚都可用作一个通用IO,或连接一个内部的外设信号。IO_MUX ¹、RTC IO MUX 和GPIO交换矩阵用于将信号从外设传输至GPIO管脚。
在本文中,先解释 MSB(最高有效位)和 LSB(最低有效位)的概念,以及 MSBFIRST 和 LSBFIRST。然后展示了 MSBFIRST 和 LSBFIRST 的使用如何影响移位寄存器的输出。
ESP32Encoder库是一个利用ESP32脉冲计数器硬件外设实现高效旋转编码器读取的软件库。
本文对比了几款适合物联网开发的盒子硬件参数,供大家参考。
乐动掌控采用掌控板作为主控,塑胶一体式外壳,侧面和底面开具多个乐高扩展孔位,兼容乐高积木,可完成多种创意应用。