小鹏浇花套件单机版程序V1.0.0

本程序是小鹏物联网智能浇花套件的单机版程序(不连接物联网),供同学们参考。

需求

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小时
  }
  
} 
- 本文为本站原创文章,转载请保留出处。
- 文章链接:https://www.xpstem.com/article/2000406

2025-08   阅读(396)   评论(0)
 标签: 创客 Arduino ESP32 FreeRTOS

涨知识
伺服电机

伺服电机(servo motor )是指在伺服系统中控制机械元件运转的发动机,是一种补助马达间接变速装置。

评论:
相关文章
【ESP32 C++教程】Unit3-2 触摸输入

本小节讲解ESP32内置触摸引脚的用法,


【ESP32 C++教程】Unit3-1 按键输入

本小节主要介绍按键信号转换、Button类及派生类、和Button交互推荐流程。


【ESP32 C++教程】Unit2-2 Ws2812灯珠

本小节主要介绍Ws2812灯珠的使用、对父类进行扩展实现自定义功能,和指针向下强制转换的使用。


【ESP32 C++教程】Unit2-1 RGB三色LED

本小节主要介绍RGB三色LED的使用,以及多态的具体实现。


【ESP32 C++教程】Unit1-3 ESP32 Arduino 开发框架

ESP32 Arduino Framework是专门针对ESP32开发板的Arduino应用开发框架,为用户开发IOT应用、HMI应用提供一致的开发体验。


ESPConnect:基于浏览器的ESP32管理器

ESPConnect是一个基于现代浏览器的管理器,在你需要快速验证、调试、管理文件、检查状态的时候,它能帮你省下大量打开和切换重型工具的时间。


用ESP32做一个FM收音机

本文介绍两种使用TEA5767收音机模块实现FM收音机的方案,感兴趣的朋友可在此基础上实现更丰富的功能。


GPIOViewer:让ESP32引脚状态一览无余!

GPIOViewer 是一个强大的 Arduino 库,专门为 ESP32 芯片设计,可以实时监控 ESP32 芯片上的所有 GPIO 引脚状态。它可以帮助你快速直观地了解每个引脚的当前状态,例如高电平、低电平、输入、输出、中断等等。


ESP32 I2S音频:初识I2S通信与配置基础

在音频处理领域,I2S是一种广泛使用的通信协议,它专门用于芯片之间的音频数据传输。ESP32 作为一款高性能的微控制器,不仅支持 I2S 通信,还提供了强大的硬件接口和灵活的软件库,使其成为音频项目开发的理想选择。


小鹏物联网自动浇花套件

小鹏物联网智能浇花系统是照顾植物的好帮手,支持自动控制和手动控制两种模式,可通过电脑端和手机端查看数据和控制浇水。