Uh oh!
There was an error while loading.Please reload this page.
- Notifications
You must be signed in to change notification settings - Fork16
Arduino library for DHT11 and DHT22 with automatic sensor recognition
License
RobTillaart/DHTNew
Folders and files
| Name | Name | Last commit message | Last commit date | |
|---|---|---|---|---|
Repository files navigation
Arduino library for DHT11 and DHT22 (and compatible) with automatic sensor type recognition.
DHTNEW is stable for both ARM and AVR. It is based upon the well tested DHTlib code.This is the main development library of all my DHT libraries.
Supports DHT11, DHT22, DHT33, DHT44, AM2301, AM2302, AM2303 as these all have the same protocol.Note there are differences e.g. DHT11 has no negative temperature, no decimals, and a longer wakeup time.
The DHTNew library returns Temperature in degrees Celsius and Humidity in 0.0 - 100.0 %RH.For converting temperature to Fahrenheit or Kelvin, seehttps://github.com/RobTillaart/Temperature.
For diagnosis if a DHT sensor gives strange readings one can usedhtnew_pulse_diag_ext.inoto compare timing with the datasheet.
Since 0.4.14 there isexperimental support for the Sonoff Si7021.No hardware yet to test this myself, but it is confirmed to work.See#79.Seems the Sonoff Si7021 sensor is very sensitive in the wakeup timing.This behaviour needs to be investigated in the future.
To use the library one should callsetType(70).
Feedback (both positive and negative) about the Sonoff Si7021 sensors is welcome.
Since 0.4.18 there isexperimental support for the AM2320, AM2321 and AM2322.Not tested myself, but AM2320 is confirmed to work, seeRobTillaart/AM232X#26As the AM2321 and AM2322 are quite identical according to the datasheet, those are expected to work too.
To use the library one should callsetType(22) as the protocol is identical to the DHT22.If there are differences in operation type (23) will be elaborated.The type 23 is now mapped upon type 22.
Feedback (both positive and negative) about the AM232X sensors is welcome.
Note: check the datasheet how to connect!
Since 0.5.1 there is support for the KY-015. Although it is a DHT11, the KY-015 behaves slightlydifferent (faster wakeup) and was recognized as a DHT22 resulting in faulty conversions.The library sensor recognition code has been adapted so it should be recognized now.In the case the recognition fails, one can usesetType(11) to force the type.
The KY-015 is confirmed to work, see#102
Feedback about the KY-015 sensors is welcome.
Note: check the datasheet how to connect!
Apparently there are DHT22's which use another representation for negative temperatures.Since 0.5.0 the library automatically detects which representation is used by the sensorand chooses the correct algorithm to decode the negative temperature.
See issue #100 (solution) and #57 and #52 before.
See alsohttps://arduino.stackexchange.com/questions/86448/dht22-sensor-reading-code-interprets-negative-values-weirdly
- https://github.com/RobTillaart/DHTNew DHT11/22 etc
- https://github.com/RobTillaart/DHTStable DHT11/22 etc
- https://github.com/RobTillaart/DHT_Simulator
- https://github.com/RobTillaart/DS18B20_INT OneWire temperature sensor
- https://github.com/RobTillaart/DS18B20_RT OneWire temperature sensor
- https://github.com/RobTillaart/SHT31 Sensirion humidity / temperature sensor
- https://github.com/RobTillaart/SHT85 Sensirion humidity / temperature sensor
- https://www.kandrsmith.org/RJS/Misc/Hygrometers/calib_many.html (interesting)
- https://github.com/RobTillaart/Temperature (conversions, dewPoint, heat index etc.)
| Front | Description | |
|---|---|---|
| pin 1 | VCC | |
| pin 2 | DATA | |
| pin 3 | Not Connected | |
| pin 4 | GND |
Note: check the datasheet of the sensor how to connect! as some have only 3 pins.
As most used model, for other sensors I refer to datasheet.
| Model | DHT22 | Notes |
|---|---|---|
| Power supply | 3.3 - 6.0 V DC | |
| Output signal | digital signal via single-bus | |
| Sensing element | polymer capacitor | |
| Operating range | humidity 0.0-100.0% RH | temperature -40° - 80° Celsius |
| Accuracy humidity | ±2% RH(Max ±5% RH) | temperature < ±0.5° Celsius |
| Resolution or sensitivity | humidity 0.1% RH | temperature 0.1° Celsius |
| Repeatability humidity | ±1.0% RH | temperature ±0.2° Celsius |
| Humidity hysteresis | ±0.3% RH | |
| Long-term Stability | ±0.5% RH/year | |
| Sensing period | average: 2 s | |
| Interchangeability | fully interchangeable | |
| Dimensions | small 14 x 18 x 5.5 mm | big 22 x 28 x 5 mm |
#include"dhtnew.h"
- DHTNEW(uint8_t pin) defines the dataPin of the sensor.
- void reset() might help to reset a sensor behaving badly.It resets the library internal settings to default,however it does not reset the sensor in a hardware way.
- uint8_t getType() 0 = unknown, 11 or 22.In case of 0,getType() will try to determine type.Since 0.4.14 type 70 is added forexperimental Sonoff Si7021 support.
- void setType(uint8_t type = 0) allows to force the type of the sensor.
| Type | Sensors | Notes |
|---|---|---|
| 0 | not defined | |
| 11 | DHT11 DHT12, KY015 | KY015 needs setType(11) |
| 22 | DHT22, DHT33, DHT44 a.o | most others |
| 23 | DHT23 | mapped to 22 for now |
| 70 | Sonoff Si7021 | experimental |
| other | sets to 0 | 0.4.20 |
- int read() reads a new temperature (Celsius) and humidity (%RH) from the sensor.
- uint32_t lastRead() returns milliseconds since lastread()
- float getHumidity() returns last read humidity = 0.0 - 100.0 %RH.In case of an error it returnsDHTLIB_INVALID_VALUE == -999.Note this error value can be suppressed bysetSuppressError(bool).
- float getTemperature() returns last read temperature in Celsius.Range depends on the sensor.In case of an error it returnsDHTLIB_INVALID_VALUE == -999.Note this error value can be suppressed bysetSuppressError(bool).
Adding offsets works well in normal range however they might introduceunder- or overflow at the ends of the sensor range.
Humidity offset is in % RH and is constrained to 0.0 - 100.0 % in the code.
Temperature offset is in degrees Celsius.For temperature a constrain would be type dependant, so it is not done.Furthermore by setting the offset to -273.15 one get theKelvin scale.
- void setHumidityOffset(float offset) typical < ±5% RH.
- void setTemperatureOffset(float offset) typical < ±2°C.
- float getHumidityOffset() idem.
- float getTemperatureOffset() idem.
The "short-named" offset functions will become obsolete in the future (0.6.0).
- void setHumOffset(float offset) typical < ±5% RH.
- void setTempOffset(float offset) typical < ±2°C.
- float getHumOffset() idem.
- float getTempOffset() idem.
Functions to adjust the communication with the sensor.
- void setDisableIRQ(bool b ) allows or suppresses interrupts during coreread function to keep timing as correct as possible.Note AVR + MKR1010 + Arduino R4
- bool getDisableIRQ() returns the above setting. Defaulttrue.
- void setWaitForReading(bool b ) flag to enforce a blocking wait.
- bool getWaitForReading() returns the above setting.
- void setReadDelay(uint16_t rd = 0) To tune the time it waits before actual read.This reduces the blocking time.Default depends on type. 1000 ms (dht11) or 2000 ms (dht22).set readDelay to 0 will reset to datasheet values AFTER a call toread().
- uint16_t getReadDelay() returns the above setting.
- void powerDown() pulls dataPin down to reduce power consumption
- void powerUp() restarts the sensor, note one must wait up to two seconds.
- void setSuppressError(bool b) suppress error values of -999 =>You need to check the return value of read() instead.
This is used to keep spikes out of your plotter / graphs / logs. - bool getSuppressError() returns the above setting.
See examples
If consistent TIMOUT_C or TIMEOUT_D occur during reading a sensor,one could try if allowing interrupts solves the issueDHT.setDisableIRQ(false).
This solved this problem at least on
- AVR boards - is build into the constructor
- MKR1010 Wifi - see#67(added as comment in the examples)
- Arduino R4 - seearduino/uno-r4-library-compatibility#38
In version 0.4.10 the TIMEOUT_C is extended from 70-90 us to even suppress the TIMEOUT_Ceven more. See discussion and tests in#67.
The MKR1010Wifi board need to wait for Serial at startup if you want to monitor itfrom the IDE. Adding the linewhile(!Serial): fixes this. (added to the examples).
There might be more boards that need this line to work properly.
- The DHT22 sensor has some problems in combination with specific pins of the ESP8266. See more details
- #31 (message Jan 3, 2021)
- arendst/Tasmota#3522
In a test an AM2301 had problems giving no humidity (99.9% overflow) when theDHTStable library was used with an ESP8266. (Reported by mail, no GH issue).As this DHTStable library is strongly related to the DHTNew it is mentioned here too.
After days of testing and thinking and more testing the cause was found.The AM2301 was powered by a 5V3 power supply which was apparently too high while having thedata handshakes at 3V3.When the VCC voltage was lowered to 5V1 it appeared to work as it should.(Kudos to Viktor for finding the cause)
DHTNEW has some new features compared to the DHTlib code.
- The constructor has a pin number, so the one sensor - one object paradigm is chosen.So you can now make a DHTNEW object bathroom(4), kitchen(3), etc.
- Theread() function now reads both DHT11 and DHT22 sensors and selects the rightmath per sensor based upon the bit patterns.
- Anoffset can be set for both temperature and humidity to have a first-order linearcalibration in the class itself. Of course, this introduces a possible risk ofunder- or overflow.For a more elaborated or non-linear offset, I refer to my multimap class.
- lastRead() keeps track of the last time the sensor is read. If this is not too long agoone can decide not to read the sensors but use the current values for temperature and humidity.This saves up to 20+ milliseconds for a DHT11 or 5+ milliseconds for a DHT22. Note that these sensorsshould have 1-2 seconds between reads according to specification.In the future, this functionality could be inside the library by setting a time threshold(e.g. 1 second by default) to give more stable results.
- Addedinterrupt enable/disable flag to prevent interrupts disturb timing of DHT protocol.Be aware that this may affect other parts of your application.
- (0.1.7) added an automatic check of lastRead in the read call. If request a read to fast it will just return OK.
- (0.1.7) addedwaitForReading flag (kudos to Mr-HaleYa) to let the sensor explicitlywait until a new value can be read.
- (0.2.0) Temperature and humidity are private now, usegetTemperature() andgetHumidity()
- (0.2.1) Adjusted the bit timing threshold to work around issue #11
- (0.2.2) addedERROR_SENSOR_NOT_READY and differentiated timeout errors.
- (0.3.0)removed interrupt flag, now the library always disables interrupts duringthe clocking of the bits.Added getReadDelay & setReadDelay to tune reading interval. Check the example code.Adjusted the timing in the wake-up part of the protocol.Added more comments to describe the protocol.
- (0.3.1)addedpowerDown() andpowerUp() for low power applications. Note that afterpowerUp()the user must wait for two seconds before doing a read(). Just like after a (re)boot.
Note: The lib does not (yet) control the power pin of the sensor.Discussion see#13 - (0.3.2)AddedsetSuppressError() andgetSuppressError() so the library will not output -999but the last known valid value for temperature and humidity.This flag is useful to suppress 'negative spikes' in graphs or logs.Default the error values are not suppressed to be backwards compatible.
Added#ifndef aroundDHTLIB_INVALID_VALUE so the default -999 can be overruledcompile time to set another error value e.g. -127 or -1 whatever suits the project. - (0.3.3)Refactored the low levelreadSensor() as theBIT SHIFT ERROR issue #29 and issue #11 popped up again.It was reproduced "efficiently" with an ESP32 and by using long wires.Fixed with an explicit digitalWrite(dataPin, HIGH) + delayMicroseconds() to have enough time betweenpulling the line HIGH and polling for the line LOW.
- (0.3.4)AddedwaitFor(state, timeout) to more precisely follow the datasheet in terms of timing.Reintroduced theinterrupt enable/disable flag as forced noInterrupts()could break the timing of the DHT protocol / micros() - seen on AVR.
- (0.4.0)AddedDHTLIB_WAITING_FOR_READ as return value of read => minor break of interface
- (0.4.1)Added Arduino-CI support +gettype() now tries to determine type if not known.
- (0.4.2)Fix negative temperatures. Tested with DHTNew_debug.ino and hex dump in .cpp and a freezer.
Note: testing in a freezer is not so good for humidity readings. - (0.4.3)Addedreset() to reset internal variables when a sensor blocks this might help.AddedlastRead() to return time the sensor is last read. (in milliseconds).
- (0.4.4)DO NOT USE incorrect negative temp.
- (0.4.5)Prevent -0.0 when negative temp is 0;DO NOT USE as it maps every negative temp to zero.
- (0.4.6)Fixed negative temperature (again).
- (0.4.7)fix #60 negative temperatures below -25.5°C + readme.md.
- (0.4.8)fixes to improve Arduino-lint.
- (0.4.9)add optional flag DHTLIB_VALUE_OUT_OF_RANGE.
- (0.4.10)updated build-CI to do compile test - UNO, due, zero, Leonardo, m4, esp32, esp8266, mega2560.updated readme.md - added badges and remarks after testing with MKR1010 Wifi.updated TIMEOUT_C from 70 -> 90 us to minimize its occurrence - See#67.added
while(!Serial);in examples to they work for MKR1010 Wifi. - (0.4.11)update library.json, license, minor edits (clean up), unit tests
- (0.4.12)Fix #72, delayMicroseconds() for wakeUp
- (0.4.13)Fix #76, disable interrupts for ESP32.
- (0.4.14)added experimental support for Si7021.
- (0.4.15)Fix #81, recognize DHT22 as type 70. Add minimal wakeup delay.
- (0.4.16)Fix #84 correct the reading of type 70 for Sonoff Si7021.
- (0.4.17)fix #86, define constants explicit as float.
- (0.4.18)Update readme.md and library.* about support for AM2320/21/22.
- (0.4.19)Update readme.md
- (0.4.20)Update GitHub actions and readme.md
- (0.4.21)Add dhtnew_pulse_diag_ext.ino
- (0.5.0)Fix negative values
- (0.5.1)Support KY015 and more
- (0.5.2)Support KY015 (again)AddDHT_endless_debug.ino develop example
- (0.5.3)Update readme.md for Arduino R4
- update documentation
- test on more boards
- #if defined(MKR1010) // TODO find out real define#67
#ifdef ARDUINO_SAMD_MKRWIFI1010#error found#endif
- test compatibility => table.
- investigate temperature constraining (type dependant)
if (type ==11) temp = constrain(temp,0,100);if (type ==22) temp = constrain(temp, -40,80);etc.
- type parameter in constructor, default 0
- reimplement the recognition algorithm (separate class?)
- read as 22 => check the hum / temp data range to determine type
- read as 11 => check idem
- read as 70 => check idem
- split wakeup time from bit interpretation used.
- move all code from .h to .cpp
- derived classes for fixed type?
If you appreciate my libraries, you can support the development and maintenance.Improve the quality of the libraries by providing issues and Pull Requests, ordonate through PayPal or GitHub sponsors.
Thank you,
About
Arduino library for DHT11 and DHT22 with automatic sensor recognition
Topics
Resources
License
Uh oh!
There was an error while loading.Please reload this page.
Stars
Watchers
Forks
Sponsor this project
Uh oh!
There was an error while loading.Please reload this page.
Packages0
Uh oh!
There was an error while loading.Please reload this page.