Arduino:Примеры/DS18x20 Temperature
Материал из Онлайн справочника
Перейти к навигацииПерейти к поиску
Содержание | Знакомство с Arduino | Продукты | Основы | Справочник языка Arduino | Примеры | Библиотеки | Хакинг | Изменения | Сравнение языков Arduino и Processing |
Перевод: Максим Кузьмин
Проверка/Оформление/Редактирование: Мякишев Е.А.
Чтение данных с температурного датчика[1]
Этот пример демонстрирует, как при помощи библиотеки OneWire считывать данные с температурных датчиков типа 1-Wire. Поддерживаются модели DS18S20, DS18B20 и DS1822.
Код
#include <OneWire.h>
// Чтение данных с температурных датчиков типа 1-Wire: DS18S20, DS18B20, DS1822
//
// http://www.pjrc.com/teensy/td_libs_OneWire.html
//
// Библиотека DallasTemperature может сделать это все за вас!
// http://milesburton.com/Dallas_Temperature_Control_Library
OneWire ds(10); // на 10-ом контакте (если необходимо, подключите резистор на 4,7 кОм)
void setup(void) {
Serial.begin(9600);
}
void loop(void) {
byte i;
byte present = 0;
byte type_s;
byte data[12];
byte addr[8];
float celsius, fahrenheit;
if ( !ds.search(addr)) {
Serial.println("No more addresses."); // "Адресов больше нет."
Serial.println();
ds.reset_search();
delay(250);
return;
}
Serial.print("ROM =");
for( i = 0; i < 8; i++) {
Serial.write(' ');
Serial.print(addr[i], HEX);
}
if (OneWire::crc8(addr, 7) != addr[7]) {
Serial.println("CRC is not valid!"); // "CRC не корректен!"
return;
}
Serial.println();
// первый ROM-байт, определяет, с каким чипом мы имеем дело:
switch (addr[0]) {
case 0x10:
Serial.println(" Chip = DS18S20"); // " Чип – DS18S20" или старый DS1820
type_s = 1;
break;
case 0x28:
Serial.println(" Chip = DS18B20"); // " Чип – DS18B20"
type_s = 0;
break;
case 0x22:
Serial.println(" Chip = DS1822"); // " Чип – DS1822"
type_s = 0;
break;
default:
Serial.println("Device is not a DS18x20 family device."); // "Устройство не принадлежит семейству DS18x20."
return;
}
ds.reset();
ds.select(addr);
ds.write(0x44, 1); // запускаем конверсию и включаем паразитное питание
delay(1000); // 750 миллисекунд может хватить, а может и нет;
// здесь можно использовать ds.depower(),
// но об этом позаботится сброс
present = ds.reset();
ds.select(addr);
ds.write(0xBE); // считываем scratchpad-память
Serial.print(" Data = "); // " Данные = "
Serial.print(present, HEX);
Serial.print(" ");
for ( i = 0; i < 9; i++) { // нам нужно 9 байтов
data[i] = ds.read();
Serial.print(data[i], HEX);
Serial.print(" ");
}
Serial.print(" CRC=");
Serial.print(OneWire::crc8(data, 8), HEX);
Serial.println();
// конвертируем данные в температуру; поскольку результат - это
// 16-битное целое знаковое число, оно должно быть записано в типе
// данных "int16_t", который всегда будет 16-битным – даже если
// данные скомпилированы на 32-битном процессоре.
int16_t raw = (data[1] << 8) | data[0];
if (type_s) {
raw = raw << 3; // разрешение по умолчанию – 9 бит
if (data[7] == 0x10) {
// регистр "count remain" дает полное 12-битное разрешение
raw = (raw & 0xFFF0) + 12 - data[6];
}
} else {
byte cfg = (data[4] & 0x60);
// при низком разрешении младшие биты не определяются, поэтому обнуляем их:
if (cfg == 0x00) raw = raw & ~7; // 9-битное разрешение (93,75 миллисекунд)
else if (cfg == 0x20) raw = raw & ~3; // 10-битное разрешение (187,5 миллисекунд)
else if (cfg == 0x40) raw = raw & ~1; // 11-битное разрешение (375 миллисекунд)
//// по умолчанию стоит 12-битное разрешение; время конверсии – 750 миллисекунд
}
celsius = (float)raw / 16.0;
fahrenheit = celsius * 1.8 + 32.0;
Serial.print(" Temperature = "); // "Температура = "
Serial.print(celsius);
Serial.print(" Celsius, "); // " по Цельсию, "
Serial.print(fahrenheit);
Serial.println(" Fahrenheit"); // " по Фаренгейту"
}