與DS18B20一樣DHT11也是采用單總線,但所不同的是DHT11可同時實現(xiàn)溫度和濕度的檢測。在我們的產(chǎn)品中經(jīng)常使用它來檢測環(huán)境的溫濕度信息。這一篇我們將設(shè)計并封裝DHT11的驅(qū)動程序,以方便重復(fù)使用。
1 、功能概述
DHT11數(shù)字溫濕度傳感器是一款含有已校準數(shù)字信號輸出的溫濕度復(fù)合傳感器。它應(yīng)用專用的數(shù)字模塊采集技術(shù)和溫濕度傳感技術(shù),確保產(chǎn)品具有極高的可靠性與卓越的長期穩(wěn)定性。
1.1 、硬件描述
傳感器包括一個電阻式感濕元件和一個NTC測溫元件,并與一個高性能8位單片機相連接。因此該產(chǎn)品具有品質(zhì)卓越、超快響應(yīng)、抗干擾能力強、性價比極高等優(yōu)點。每個DHT11傳感器都在極為精確的濕度校驗室中進行校準。校準系數(shù)以程序的形式儲存在OTP內(nèi)存中,傳感器內(nèi)部在檢測信號的處理過程中要調(diào)用這些校準系數(shù)。單線制串行接口,使系統(tǒng)集成變得簡易快捷。超小的體積、極低的功耗,信號傳輸距離可達20米以上,使其成為各類應(yīng)用甚至最為苛刻的應(yīng)用場合的最佳選擇。產(chǎn)品為4針單排引腳封裝。
DHT11的供電電壓為 3-5.5V。傳感器上電后,要等待 1s 以越過不穩(wěn)定狀態(tài)在此期間無需發(fā)送任何指令。電源引腳(VDD,GND)之間可增加一個100nF 的電容,用以去耦濾波。
1.2 、通訊接口
DHT11傳感器單總線通訊建議連接線長度短于20米時用5K上拉電阻,大于20米時根據(jù)實際情況使用合適的上拉電阻。連線圖如下:
DATA用于微處理器與DHT11之間的通訊和同步,采用單總線數(shù)據(jù)格式,一次通訊時間4ms左右,數(shù)據(jù)分小數(shù)部分和整數(shù)部分,具體格式在下面說明,當前小數(shù)部分用于以后擴展,現(xiàn)讀出為零。一次完整的數(shù)據(jù)傳輸為40bit,高位在前。數(shù)據(jù)格式:8bit濕度整數(shù)數(shù)據(jù)+8bit濕度小數(shù)數(shù)據(jù)+8bi溫度整數(shù)數(shù)據(jù)+8bit溫度小數(shù)數(shù)據(jù)+8bit校驗和。
其中,在數(shù)據(jù)傳送正確時,校驗和數(shù)據(jù)等于“8bit濕度整數(shù)數(shù)據(jù)+8bit濕度小數(shù)數(shù)據(jù)+8bi溫度整數(shù)數(shù)據(jù)+8bit溫度小數(shù)數(shù)據(jù)”所得結(jié)果的末8位。
用戶MCU發(fā)送一次開始信號后,DHT11從低功耗模式轉(zhuǎn)換到高速模式,等待主機開始信號結(jié)束后,DHT11發(fā)送響應(yīng)信號,送出40bit的數(shù)據(jù),并觸發(fā)一次信號采集,用戶可選擇讀取部分數(shù)據(jù)。通訊過程如下圖所示:
從模式下,DHT11接收到開始信號觸發(fā)一次溫濕度采集,如果沒有接收到主機發(fā)送開始信號,DHT11不會主動進行溫濕度采集。采集數(shù)據(jù)后轉(zhuǎn)換到低速模式。
2 、驅(qū)動設(shè)計與實現(xiàn)
我們已經(jīng)了解了DHT11溫濕度傳感器的相關(guān)信息,接下來我們將設(shè)計并實現(xiàn)DHT11溫濕度傳感器的驅(qū)動程序。
2.1 、對象定義
在使用一個對象之前我們需要獲得這個對象。同樣的我們想要操作DHT11溫濕度傳感器就需要先定義DHT11溫濕度傳感器的對象。
2.1.1 、對象的抽象
我們要得到DHT11溫濕度傳感器對象,需要先分析其基本特性。一般來說,一個對象至少包含兩方面的特性:屬性與操作。接下來我們就來從這兩個方面思考一下DHT11溫濕度傳感器的對象。
先來考慮屬性,作為屬性肯定是用于標識或記錄對象特征的東西。我們來考慮DHT11溫濕度傳感器對象屬性。DHT11溫濕度傳感器并沒有標識設(shè)備區(qū)別的特性,只有溫度和濕度信息可以表示當前的工作狀態(tài)我們將其作為屬性。
接著我們還需要考慮DHT11溫濕度傳感器對象的操作問題。我們知道DHT11溫濕度傳感器采用的是單總線。單總線就需要控制總線的輸入輸出方向,而且這對這條總線在不同的輸入輸出方向,我們需要讀數(shù)據(jù)和寫數(shù)據(jù),而這些操作都依賴于硬件平臺,所以我們將他們定義為DHT11溫濕度傳感器對象的操作。處于時序控制的需要,我們需要延時操作函數(shù),而在不同的軟硬件平臺延時操作會有差異,我們也將其作為對象的操作。
根據(jù)上述我們對DHT11溫濕度傳感器的分析,我們可以定義DHT11溫濕度傳感器的對象類型如下:
1 /* 定義DHT11對象類型 */
2 typedef struct Dht11Object {
3 float temperature; //溫度值
4 float humidity; //濕度值
5
6 uint8_t (*SetPinOutValue)(DhtPinValueType setValue);//設(shè)置DHT11引腳的輸出值
7 uint8_t (*ReadPinBit)(void);//讀取引腳電平
8 void (*SetPinDirection)(DHT11IOModeType mode);//設(shè)置引腳的輸入輸出方向
9
10 void (*Delayms)(volatile uint32_t nTime); /*實現(xiàn)ms延時操作*/
11 void (*Delayus)(volatile uint32_t nTime); /*實現(xiàn)us延時操作*/
12 }Dht11ObjectType;
2.1.2 、對象初始化
我們知道,一個對象僅作聲明是不能使用的,我們需要先對其進行初始化,所以這里我們來考慮DHT11溫濕度傳感器對象的初始化函數(shù)。一般來說,初始化函數(shù)需要處理幾個方面的問題。一是檢查輸入參數(shù)是否合理;二是為對象的屬性賦初值;三是對對象作必要的初始化配置。據(jù)此我們設(shè)計DHT11溫濕度傳感器對象的初始化函數(shù)如下:
1 /*DHT11初始化操作*/
2 DHT11ErrorType InitializeDHT11(Dht11ObjectType *dht, //需要初始化對象
3 Dht11SetPinOutValueType setPinStatus, //設(shè)置總線輸出值
4 Dht11ReadPinBitType getPinStatus, //讀取總線輸入值
5 Dht11SetPinModeType mode, //配置總線的輸入輸出模式
6 Dht11DelayType delayms, //毫秒延時
7 Dht11DelayType delayus //微秒延時
8 )
9 {
10 if((dht==NULL)||(setPinStatus==NULL)||(getPinStatus==NULL)||(mode==NULL)||(delayms==NULL)||(delayus==NULL))
11 {
12 return DHT11_InitError;
13 }
14
15 dht->SetPinOutValue=setPinStatus;
16 dht->ReadPinBit=getPinStatus;
17 dht->SetPinMode=mode;
18 dht->Delayms=delayms;
19 dht->Delayus=delayus;
20
21 dht->humidity=0.0;
22 dht->temperature=0.0;
23
24 ResetDHT11(dht);
25 return CheckDHT11Status(dht);
26 }
2.2 、對象操作
我們已經(jīng)完成了DHT11溫濕度傳感器對象類型的定義和對象初始化函數(shù)的設(shè)計。但我們的主要目標是獲取對象的信息,接下來我們還要實現(xiàn)面向DHT11溫濕度傳感器的各類操作。
2.2.1 、啟動數(shù)據(jù)通訊
DHT11溫濕度傳感器上電后,總線空閑狀態(tài)為高電平,主機把總線拉低等待DHT11響應(yīng),主機把總線拉低必須大于18毫秒,保證DHT11能檢測到起始信號。DHT11接收到主機的開始信號后,等待主機開始信號結(jié)束,然后發(fā)送80us低電平響應(yīng)信號。主機發(fā)送開始信號結(jié)束后,延時等待20-40us后, 讀取DHT11的響應(yīng)信號,主機發(fā)送開始信號后,可以切換到輸入模式,或者輸出高電平均可, 總線由上拉電阻拉高。啟動數(shù)據(jù)通訊的時序圖如下:
1 /*復(fù)位DHT11,開始通訊*/
2 static void ResetDHT11(Dht11ObjectType *dht)
3 {
4 dht->SetPinMode(DHT11_Out); //設(shè)置為輸出方式
5 dht->SetPinOutValue(DHT11_Reset); //將引腳點位拉低
6 dht->Delayms(20); //拉低至少18ms
7 dht->SetPinOutValue(DHT11_Set); //拉高
8 dht->Delayus(30); //主機拉高20至40us
9 }
DHT11傳感器的DATA引腳檢測到外部信號有低電平時,等待外部信號低電平結(jié)束,延遲后DHT11的 DATA引腳處于輸出狀態(tài),輸出80微秒的低電平作為應(yīng)答信號,緊接著輸出 80 微秒的高電平通知外設(shè)準備接收數(shù)據(jù),微處理器的 I/O 此時處于輸入狀態(tài),檢測到 I/O 有低電平(DHT11 回應(yīng)信號)后,等待80微秒的高電平后的數(shù)據(jù)接收。
1 /*等待DHT11的回應(yīng),返回1:未檢測到DHT11的存在;返回0:存在*/
2 static DHT11ErrorType CheckDHT11Status(Dht11ObjectType *dht)
3 {
4 uint8_t retry=0;
5 dht->SetPinMode(DHT11_In); //設(shè)置為輸入方式
6 while(dht->ReadPinBit()&&(retry<100))
7 {
8 retry++;
9 dht->Delayus(1);
10 }
11 if(retry>=100)
12 {
13 return DHT11_None;
14 }
15 retry=0;
16 while(!dht->ReadPinBit()&&(retry<100))
17 {
18 retry++;
19 dht->Delayus(1);
20 }
21 if(retry>=100)
22 {
23 return DHT11_None;
24 }
25 return DHT11_NoError;
26 }
2.2.2 、讀取數(shù)據(jù)位
當主機變?yōu)檩斎肽J胶?,檢測到總線為低電平,說明DHT11發(fā)送響應(yīng)信號,DHT11發(fā)送響應(yīng)信號后,再把總線拉高80us,準備發(fā)送數(shù)據(jù),每一bit數(shù)據(jù)都以50us低電平時隙開始,高電平的長短定了數(shù)據(jù)位是“0”還是“1”。表示0”和“1”的時序圖如下所示:
1 /*從DHT11讀取一個位,返回值:1/0*/
2 static uint8_t ReadBitFromDHT11(Dht11ObjectType *dht)
3 {
4 uint8_t retry=0;
5 /*等待變?yōu)榈碗娖?/
6 while(dht->ReadPinBit()&&(retry<100))
7 {
8 retry++;
9 dht->Delayus(1);
10 }
11 retry=0;
12 /*等待變高電平*/
13 while(!dht->ReadPinBit()&&(retry<100))
14 {
15 retry++;
16 dht->Delayus(1);
17 }
18 dht->Delayus(40); //延時判斷此位是0還是1
19
20 return dht->ReadPinBit();
21 }
當最后一bit數(shù)據(jù)傳送完畢后,DHT11拉低總線50us,隨后總線由上拉電阻拉高進入空閑狀態(tài)。
3 、驅(qū)動的使用
我們已經(jīng)實現(xiàn)了DHT11溫濕度傳感器的驅(qū)動,接下來將以此驅(qū)動為基礎(chǔ)設(shè)計了簡單的測試應(yīng)用。
3.1 、聲明并初始化對象
使用基于對象的操作我們需要先得到這個對象,所以我們先要使用前面定義的DHT11溫濕度傳感器對象類型聲明一個DHT11溫濕度傳感器對象變量,具體操作格式如下:
Dht11ObjectType dht;
聲明了這個對象變量并不能立即使用,我們還需要使用驅(qū)動中定義的初始化函數(shù)對這個變量進行初始化。這個初始化函數(shù)所需要的輸入?yún)?shù)如下:
Dht11ObjectType *dht,需要初始化對象
Dht11SetPinOutValueType setPinStatus,設(shè)置總線輸出值
Dht11ReadPinBitType getPinStatus,讀取總線輸入值
Dht11SetPinModeType mode,配置總線的輸入輸出模式
Dht11DelayType delayms,毫秒延時
Dht11DelayType delayus,微秒延時
對于這些參數(shù),對象變量我們已經(jīng)定義了。剩下的輸入?yún)?shù)就是我們操作中需要的函數(shù),這幾個函數(shù)需要我們在應(yīng)用中定義,并將函數(shù)指針作為參數(shù)。這幾個函數(shù)的類型如下:
1 typedef uint8_t (*Dht11SetPinOutValueType)(DhtPinValueType setValue);//設(shè)置DHT11引腳的輸出值
2 typedef uint8_t (*Dht11ReadPinBitType)(void);//讀取引腳電平
3 typedef void (*Dht11SetPinModeType)(DHT11IOModeType mode);//設(shè)置引腳的輸入輸出方向
4 typedef void (*Dht11DelayType)(volatile uint32_t nTime); /*實現(xiàn)ms延時操作*/
對于這幾個函數(shù)我們根據(jù)樣式定義就可以了,具體的操作可能與使用的硬件平臺有關(guān)系。我們在STM32F4及HAL庫環(huán)境下使用,具體函數(shù)定義如下:
1 //設(shè)置DHT11引腳的輸出值
2 uint8_t Dht11SetPinOutValue(DhtPinValueType setValue)
3 {
4 HAL_GPIO_WritePin(GPIOB,GPIO_PIN_11,(GPIO_PinState)setValue);
5 }
6
7 //讀取引腳電平
8 uint8_t Dht11ReadPinBit(void)
9 {
10 return (uint8_t)HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_11);
11 }
12
13 //設(shè)置引腳的輸入輸出方向
14 void Dht11SetPinMode(DHT11IOModeType mode)
15 {
16 GPIO_InitTypeDef GPIO_InitStruct;
17
18 GPIO_InitStruct.Pin = GPIO_PIN_11;
19 if(mode==DHT11_In)
20 {
21 GPIO_InitStruct.Mode = GPIO_MODE_INPUT;
22 GPIO_InitStruct.Pull = GPIO_NOPULL;
23 }
24 else
25 {
26 GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
27 GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;
28 }
29 HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);
30 }
對于延時函數(shù)我們可以采用各種方法實現(xiàn)。我們采用的STM32平臺和HAL庫則可以直接使用HAL_Delay()函數(shù)。微秒延時函數(shù)則使用我們自己定義的。于是我們可以調(diào)用初始化函數(shù)如下:
InitializeDHT11(&dht,Dht11SetPinOutValue,Dht11ReadPinBit,Dht11SetPinMode,HAL_Delay,Delayus);
3.2 、基于對象進行操作
我們定義了對象變量并使用初始化函數(shù)給其作了初始化。接著我們就來考慮操作這一對象獲取我們想要的數(shù)據(jù)。我們在驅(qū)動中已經(jīng)將獲取數(shù)據(jù)并轉(zhuǎn)換為轉(zhuǎn)換值的比例值,接下來我們使用這一驅(qū)動開發(fā)我們的應(yīng)用實例。
1 /*獲取數(shù)據(jù)值*/
2 void GetMeasureDataFromDHT11(void)
3 {
4 float temperature; //溫度值
5 float humidity; //濕度值
6
7 GetProcessValueFromDHT11(&dht);
8
9 temperature=dht.temperature;
10 humidity=dht.humidity;
11 }
4 、應(yīng)用總結(jié)
我們已經(jīng)實現(xiàn)了DHT11溫濕度傳感器的驅(qū)動,并在此基礎(chǔ)上設(shè)計了簡單的驗證應(yīng)用。經(jīng)過測試,利用驅(qū)動我們成功的讀取了溫濕度數(shù)據(jù)。
根據(jù)數(shù)據(jù)手冊的要求,DHT11溫濕度傳感器上電后要等待1S以越過不穩(wěn)定狀態(tài)在此期間不能發(fā)送任何指令。
單總線數(shù)據(jù)傳輸時,會改變總線的輸入輸出方向。在我們的應(yīng)用中,我們修改了對應(yīng)GPIO引腳的輸入輸出模式。事實上如果我們在STM32中使用時,我們可將該引腳配置為開漏輸出模式,加上總線的上拉電阻,可以在不修改GPIO的輸入輸出模式的情況下實現(xiàn)讀寫。
源碼下載:https://github.com/foxclever/ExPeriphDriver
評論
查看更多