Применение термостата с ПИД управлением не ограничено птицеводством, КО, проект может использоваться пивоварами для поддержания температуры сусла, винокурами в перегонных аппаратах, может просто греть воду в бойлере до приятной температуры, после небольших изменений, в самодельных паяльных станциях, муфельных печах, кароче везде где требуется контроль температуры с высокой точность. Принцип работы и отличие от банального термостата с гистерезисом показан в видео:
Ниже схемы подключения и исходные коды проекта.
О датчиках температуры:
Исходные коды проекта написаны нескольких типов датчиков:
- TMP102 — датчик температуры с интерфейсом I2C, подробнее про работу с датчиком написано тут. Возможно датчик потребует корректировку температуры.
- LM35 — не дорогой и распространенный, достаточно точный аналоговый датчик температуры, подробнее про него тут.
- MCP9808 — высокоточный датчик температуры с интерфейсом I2C. Оптимальный выбор.
Имея навыки программирования ардуино, изменить код под другие датчики температуры труда не составит.
О ЖКИ индикаторе:
Решил не усложнять и взял стандартный текстовый экран WH1602A, про подключение подобных экранов к ардуино уже написано тут. Подключается напрямую, без переходников на I2C. Указанные в проекте номера выходов для подключения ЖКИ совпадают с китайским LCD Keypad Shield, я его использовал на стадии отладки.
Органы управления:
Настройка температуры терморегулятора осуществляется с помощью энкодера, удобно использовать модуль KY-040 по китайской номенклатуре 🙂
О реле:
В регуляторе крайне не рекомендуется использовать электромеханические реле, от частого переключения они выйдут из строя, особенно это относится к китайским релейным модулям за пару долларов.
Правильным решением будет использование твердотельного реле, либо модуль с симистором (что по сути является одним и тем же) на необходимый ток. Например SSR-25DA на 25А, если мощность нагревателя в районе нескольких киловатт, OMRON G3MB 202P держит до 2А или 440Вт.
У меня они в наличии не оказались, пришлось закупить в местном магазине радиодеталей оптосимистор MOC3063 для гальванической развязки и «детекции нуля» с симистором BT137x-800 на 8А и собирать твердотельное реле на макетке. Схема взята из даташита на MOC3063.
Схема подключения:
Структурная схема:
Возможно на этой будет понятней.
Схема подключения актуальная для датчиков с интерфейсом I2C (MCP9808, TMP102), в случаи использования аналогового LM35, его выход подключается к аналоговому входу А5.
Исходные коды:
//Термостат для инкубатора или PID регулятор на arduino //код из видео https://www.youtube.com/watch?v=9odcsuAquLU //донат, http://www.donationalerts.ru/r/arduinolab /// датчик температуры TMP102, LCD1602 #include <EEPROM.h> #include <Wire.h> #include <LiquidCrystal.h> #include "SparkFunTMP102.h" #include <PID_v1.h> TMP102 tmp102(0x48); LiquidCrystal lcd(8, 9, 4, 5, 6, 7); ////////////////////////////////////////// #define ENCODER_A 2 // вход энкодера #define ENCODER_B 3 #define ENCODER_KEY 11 #define TEMP_MAX 60.0 // Приделы настройки термостата #define TEMP_MIN 25.0 #define RELAY_PIN 12 // выход на реле #define WindowSize 512 // периуд, для симистора можно оставить, для // реле увеличить в ~10 раз ////////////////////////////////////////////// bool encoderPinALast = LOW; bool n = LOW; unsigned long windowStartTime; double Setpoint, Input, Output; double Kp=2, Ki=5, Kd=1.5; PID myPID(&Input, &Output, &Setpoint, Kp, Ki, Kd, DIRECT); float temperature; void setup() { // Serial.begin(9600); tmp102.begin(); lcd.begin(16, 2); lcd.clear(); pinMode(RELAY_PIN, OUTPUT); pinMode (ENCODER_A, INPUT); pinMode (ENCODER_B, INPUT); pinMode (ENCODER_KEY, INPUT); tmp102.wakeup(); myPID.SetOutputLimits(0, WindowSize); myPID.SetMode(AUTOMATIC); //// проверка ЕЕПРОМ для первого включения if (EEPROM.read(0) == 255 && EEPROM.read(4) == 255){ EEPROM.write(0, 10); EEPROM.write(1, 0); EEPROM.write(2, 240); EEPROM.write(3, 65); } EEPROM.get(0, Setpoint); } void loop() { temperature = tmp102.readTempC(); Input = temperature; myPID.Compute(); lcd.setCursor(0, 0); lcd.print(temperature); lcd.print("C "); lcd.setCursor(8, 0); lcd.print(Setpoint, 1); lcd.print("C "); lcd.setCursor(0, 1); lcd.print(map(Output, 0, WindowSize, 0, 100)); lcd.print("% "); lcd.setCursor(8, 1); lcd.print(Output); lcd.print("mS "); /////////////////////// if(!digitalRead(ENCODER_KEY)){ // если надавили на кнопку, входим в настройки lcd.setCursor(13, 0); lcd.print("<<"); // управление энкодером n = digitalRead(ENCODER_A); if ((encoderPinALast == LOW) && (n == HIGH)) { if (digitalRead(ENCODER_B) == LOW) { Setpoint -= .1; if (Setpoint < TEMP_MIN) Setpoint = TEMP_MIN; // проверяем приделы настроек } else { Setpoint += .1; if (Setpoint > TEMP_MAX) Setpoint = TEMP_MAX; } } EEPROM.put(0, Setpoint); // пишим в еепром настройку encoderPinALast = n; delay(5); // return; } /************************************************ * turn the output pin on/off based on pid output ************************************************/ if (millis() - windowStartTime >= WindowSize) { //time to shift the Relay Window windowStartTime += WindowSize; if (windowStartTime > millis()) windowStartTime = 0; // защита от переполнения } if (Output < millis() - windowStartTime) digitalWrite(RELAY_PIN, LOW); else digitalWrite(RELAY_PIN, HIGH); }
//Термостат для инкубатора или PID регулятор на arduino //код из видео https://www.youtube.com/watch?v=9odcsuAquLU //донат, http://www.donationalerts.ru/r/arduinolab /// датчик температуры TMP102, LCD0802 /// отличается только выводом информации на ЖК #include <EEPROM.h> #include <Wire.h> #include <LiquidCrystal.h> #include "SparkFunTMP102.h" // https://github.com/sparkfun/SparkFun_TMP102_Arduino_Library #include <PID_v1.h> // https://playground.arduino.cc/Code/PIDLibrary TMP102 tmp102(0x48); // адрес TMP102 на шине, может отличатся. LiquidCrystal lcd(8, 9, 4, 5, 6, 7); ////////////////////////////////////////// #define ENCODER_A 2 // вход энкодера #define ENCODER_B 3 #define ENCODER_KEY 11 #define TEMP_MAX 60.0 // Приделы настройки термостата #define TEMP_MIN 25.0 #define RELAY_PIN 12 // выход на реле #define WindowSize 512 // периуд, для симистора можно оставить, для // реле увеличить в ~10 раз ////////////////////////////////////////////// bool encoderPinALast = LOW; bool n = LOW; unsigned long windowStartTime; double Setpoint, Input, Output; double Kp=2, Ki=5, Kd=1.5; PID myPID(&Input, &Output, &Setpoint, Kp, Ki, Kd, DIRECT); float temperature; void setup() { // Serial.begin(9600); tmp102.begin(); lcd.begin(8, 2); lcd.clear(); pinMode(RELAY_PIN, OUTPUT); pinMode(ENCODER_A, INPUT); pinMode(ENCODER_B, INPUT); pinMode(ENCODER_KEY, INPUT); tmp102.wakeup(); myPID.SetOutputLimits(0, WindowSize); myPID.SetMode(AUTOMATIC); //// проверка ЕЕПРОМ для первого включения if (EEPROM.read(0) == 255 && EEPROM.read(4) == 255){ EEPROM.write(0, 10); EEPROM.write(1, 0); EEPROM.write(2, 240); EEPROM.write(3, 65); } EEPROM.get(0, Setpoint); } void loop() { temperature = tmp102.readTempC(); Input = temperature; myPID.Compute(); /////////////////////// if(!digitalRead(ENCODER_KEY)){ // если надавили на кнопку, входим в настройки lcd.setCursor(0, 0); // отображаем страницу настроек lcd.print("settings"); lcd.setCursor(1, 1); lcd.print(Setpoint, 1); lcd.print("C "); // управление энкодером n = digitalRead(ENCODER_A); if ((encoderPinALast == LOW) && (n == HIGH)) { if (digitalRead(ENCODER_B) == LOW) { Setpoint -= .1; if (Setpoint < TEMP_MIN) Setpoint = TEMP_MIN; // проверяем приделы настроек } else { Setpoint += .1; if (Setpoint > TEMP_MAX) Setpoint = TEMP_MAX; } } EEPROM.put(0, Setpoint); // пишим в еепром настройку encoderPinALast = n; delay(5); } // иначе отобрадаем страницу с текущей else{ // температурой и мощьностью lcd.setCursor(0, 0); lcd.print(temperature); lcd.print("C "); lcd.setCursor(1, 1); lcd.print(map(Output, 0, WindowSize, 0, 100)); lcd.print("% "); } /************************************************ * turn the output pin on/off based on pid output ************************************************/ if (millis() - windowStartTime >= WindowSize) { // таймер на миллис windowStartTime += WindowSize; if (windowStartTime > millis()) windowStartTime = 0; // защита от переполнения } if (Output < millis() - windowStartTime) digitalWrite(RELAY_PIN, LOW); else digitalWrite(RELAY_PIN, HIGH); }
//Термостат для инкубатора или PID регулятор на arduino //код из видео https://www.youtube.com/watch?v=9odcsuAquLU //донат, http://www.donationalerts.ru/r/arduinolab /// датчик температуры LM35 #include <EEPROM.h> #include <Wire.h> #include <LiquidCrystal.h> #include <PID_v1.h> // https://playground.arduino.cc/Code/PIDLibrary LiquidCrystal lcd(8, 9, 4, 5, 6, 7); ////////////////////////////////////////// #define ENCODER_A 2 // вход энкодера #define ENCODER_B 3 #define ENCODER_KEY 11 #define LM35_PIN A5 // вход датчика LM35 (1-5V, 2-OUT, 3-GND) #define TEMP_MAX 60.0 // Приделы настройки термостата #define TEMP_MIN 25.0 #define RELAY_PIN 12 // выход на реле #define WindowSize 500 // периуд, для симистора можно оставить, для // реле увеличить в ~10 раз ////////////////////////////////////////////// bool encoderPinALast = LOW; bool n = LOW; unsigned long windowStartTime; double Setpoint, Input, Output; double Kp=2, Ki=5, Kd=1.5; // коэффициенты настройки PID PID myPID(&Input, &Output, &Setpoint, Kp, Ki, Kd, DIRECT); float temperature; void setup() { // Serial.begin(9600); lcd.begin(16, 2); lcd.clear(); pinMode(RELAY_PIN, OUTPUT); pinMode (ENCODER_A, INPUT); pinMode (ENCODER_B, INPUT); pinMode (ENCODER_KEY, INPUT); analogReference(INTERNAL); // включаем внутрений источник опорного 1,1 вольт myPID.SetOutputLimits(0, WindowSize); myPID.SetMode(AUTOMATIC); //// проверка ЕЕПРОМ для первого включения if (EEPROM.read(0) == 255 && EEPROM.read(4) == 255){ EEPROM.write(0, 10); EEPROM.write(1, 0); EEPROM.write(2, 240); EEPROM.write(3, 65); } EEPROM.get(0, Setpoint); } void loop() { temperature = analogRead(LM35_PIN) / 9.31; // забераем температуру Input = temperature; myPID.Compute(); /////////////////////// if(!digitalRead(ENCODER_KEY)){ // если надавили на кнопку, входим в настройки digitalWrite(RELAY_PIN, LOW); // и выключаем нагреватель. lcd.clear(); // очищаем экран и выводим параметр lcd.setCursor(8, 0); lcd.print(Setpoint, 1); lcd.print("C "); // управление энкодером n = digitalRead(ENCODER_A); if ((encoderPinALast == LOW) && (n == HIGH)) { if (digitalRead(ENCODER_B) == LOW) { Setpoint -= .1; if (Setpoint < TEMP_MIN) Setpoint = TEMP_MIN; // проверяем приделы настроек } else { Setpoint += .1; if (Setpoint > TEMP_MAX) Setpoint = TEMP_MAX; } EEPROM.put(0, Setpoint); // пишим в еепром настройку } encoderPinALast = n; delay(5); return; } // if (millis() - windowStartTime >= WindowSize) { // таймер на миллис windowStartTime += WindowSize; if (windowStartTime > millis()) windowStartTime = 0; // защита от переполнения lcd.setCursor(0, 0); lcd.print(temperature); // вывод на экран перенес в таймер lcd.print("C "); // чтение с LM35 происходит быстрее. lcd.setCursor(8, 0); // отсутствует необходимая LCD lcd.print(Setpoint, 1); // экрану зарежка lcd.print("C "); lcd.setCursor(0, 1); lcd.print(map(Output, 0, WindowSize, 0, 100)); lcd.print("% "); lcd.setCursor(8, 1); lcd.print(Output); lcd.print("mS "); } if (Output < millis() - windowStartTime) digitalWrite(RELAY_PIN, LOW); else digitalWrite(RELAY_PIN, HIGH); }
//Термостат для инкубатора или PID регулятор на arduino //код из видео https://www.youtube.com/watch?v=9odcsuAquLU //донат, http://www.donationalerts.ru/r/arduinolab /// датчик температуры MCP9808, LCD1602 #include <EEPROM.h> #include <Wire.h> #include <LiquidCrystal.h> #include "Adafruit_MCP9808.h" // https://codeload.github.com/adafruit/Adafruit_MCP9808_Library/zip/master #include <PID_v1.h> // https://playground.arduino.cc/Code/PIDLibrary Adafruit_MCP9808 tempsensor = Adafruit_MCP9808(); LiquidCrystal lcd(8, 9, 4, 5, 6, 7); ////////////////////////////////////////// #define ENCODER_A 2 // вход энкодера #define ENCODER_B 3 #define ENCODER_KEY 11 #define TEMP_MAX 60.0 // Приделы настройки термостата #define TEMP_MIN 25.0 #define RELAY_PIN 12 // выход на реле #define WindowSize 512 // периуд, для симистора можно оставить, для // реле увеличить в ~10 раз ////////////////////////////////////////////// bool encoderPinALast = LOW; bool n = LOW; unsigned long windowStartTime; double Setpoint, Input, Output; double Kp=2, Ki=5, Kd=1.5; PID myPID(&Input, &Output, &Setpoint, Kp, Ki, Kd, DIRECT); float temperature; void setup() { // Serial.begin(9600); lcd.begin(16, 2); lcd.clear(); if (!tempsensor.begin()) { lcd.print("not find MCP9808!"); while (1); } pinMode(RELAY_PIN, OUTPUT); pinMode (ENCODER_A, INPUT); pinMode (ENCODER_B, INPUT); pinMode (ENCODER_KEY, INPUT); myPID.SetOutputLimits(0, WindowSize); myPID.SetMode(AUTOMATIC); //// проверка ЕЕПРОМ для первого включения if (EEPROM.read(0) == 255 && EEPROM.read(4) == 255){ EEPROM.write(0, 10); EEPROM.write(1, 0); EEPROM.write(2, 240); EEPROM.write(3, 65); } EEPROM.get(0, Setpoint); } void loop() { temperature = tempsensor.readTempC(); Input = temperature; myPID.Compute(); lcd.setCursor(0, 0); lcd.print(temperature); lcd.print("C "); lcd.setCursor(8, 0); lcd.print(Setpoint, 1); lcd.print("C "); lcd.setCursor(0, 1); lcd.print(map(Output, 0, WindowSize, 0, 100)); lcd.print("% "); lcd.setCursor(8, 1); lcd.print(Output); lcd.print("mS "); /////////////////////// if(!digitalRead(ENCODER_KEY)){ // если надавили на кнопку, входим в настройки lcd.setCursor(13, 0); lcd.print("<<"); // управление энкодером n = digitalRead(ENCODER_A); if ((encoderPinALast == LOW) && (n == HIGH)) { if (digitalRead(ENCODER_B) == LOW) { Setpoint -= .1; if (Setpoint < TEMP_MIN) Setpoint = TEMP_MIN; // проверяем приделы настроек } else { Setpoint += .1; if (Setpoint > TEMP_MAX) Setpoint = TEMP_MAX; } } EEPROM.put(0, Setpoint); // пишим в еепром настройку encoderPinALast = n; delay(5); // return; } /************************************************ * turn the output pin on/off based on pid output ************************************************/ if (millis() - windowStartTime >= WindowSize) { //time to shift the Relay Window windowStartTime += WindowSize; if (windowStartTime > millis()) windowStartTime = 0; // защита от переполнения } if (Output < millis() - windowStartTime) digitalWrite(RELAY_PIN, LOW); else digitalWrite(RELAY_PIN, HIGH); }
Возможные проблемы и пути решения:
пока всё.