?導(dǎo)讀:《藍橋杯單片機組》專欄文章是博主2018年參加藍橋杯的單片機組比賽所做的學(xué)習(xí)筆記,在當年的比賽中,博主是獲得了省賽一等獎,國賽二等獎的成績。成績雖談不上最好,但至少問心無愧。如今2021年回頭再看該系列文章,仍然感觸頗多。為了能更好地幫助到單片機初學(xué)者,今年特地抽出時間對當年的文章邏輯和結(jié)構(gòu)進行重構(gòu),以達到初學(xué)者快速上手的目的。需要指出的是,由于本人水平有限,如有錯誤還請讀者指出,非常感謝。那么,接下來讓我們一起開始愉快的學(xué)習(xí)吧。
上一節(jié)我們通過PCF8591這個期間入門學(xué)習(xí)了ADC相關(guān)知識,本篇博文我們繼續(xù)再來學(xué)習(xí)一下一個常見的溫度傳感器模塊,它使用的通信協(xié)議是DS18B20,話不多說,開搞。程序代碼可到Github下載<傳送門>。
一、基礎(chǔ)理論
注意:18B20對時序要求嚴格,不可被中斷打斷!一旦打斷會出現(xiàn)亂碼,顯示的溫度是亂碼。所以切記,操作的時候關(guān)EA !
2018年3月30日更:中午吃完飯仔細想想這個里面還是有問題的,關(guān)了中斷意味著我們的實時任務(wù)處理不了了!可能導(dǎo)致部分功能就沒法實現(xiàn)了!既然現(xiàn)在面臨的問題是:溫度顯示是亂碼。那再想一下,原因還是我們直接從18b20讀然后直接實時顯示導(dǎo)致的,那我們不實時顯示,而是放到緩沖區(qū)里面。然后對溫度進行合法性檢驗再從緩沖區(qū)里讀出顯示不也可以嘛!
注意最下面程序中的軟件延時操作技巧,IT單片機和12T單片機,關(guān)于nop延時,可以看這里。<傳送門>
這里寫圖片描述
一共 2 個字節(jié),LSB 是低字節(jié),MSB 是高字節(jié),其中 MSb 是字節(jié)的高位,LSb 是字節(jié)的低位。可以看出來,二進制數(shù)字,每一位代表的溫度的含義,都表示出來了。其中 S表示的是符號位,低 11 位都是 2 的冪,用來表示最終的溫度。DS18B20 的溫度測量范圍是從-55 度到+125 度,而溫度數(shù)據(jù)的表現(xiàn)形式,有正負溫度,寄存器中每個數(shù)字如同卡尺的刻度一樣分布。
這里寫圖片描述
二進制數(shù)字最低位變化 1,代表溫度變化 0.0625 度的映射關(guān)系。當 0 度
的時候,那就是0x0000
,當溫度 125 度
的時候,對應(yīng)十六進制是 0x07D0
,當溫度是零下 55 度
的時候,對應(yīng)的數(shù)字是 0xFC90
。反過來說,當數(shù)字是 0x0001
的時候,那溫度就是 0.0625
度了。
對應(yīng)的協(xié)議為 1-wire!!!
1.1、初始化
這里寫圖片描述
和 I 2 C 的尋址類似,1-Wire 總線開始也需要檢測這條總線上是否存在 DS18B20 這個器件。如果這條總線上存在 DS18B20,總線會根據(jù)時序要求返回一個低電平脈沖,如果不存在的話,也就不會返回脈沖,即總線保持為高電平,所以習(xí)慣上稱之為檢測存在脈沖。
存在脈沖檢測過程,首先單片機要拉低這個引腳,持續(xù)大概 480us 到 960us 之間的時間即可,我們的程序中持續(xù)了 500us。然后,單片機釋放總線,就是給高電平,DS18B20 等待大概 15 到 60us 后,會主動拉低這個引腳大概是 60 到 240us,而后 DS18B20 會主動釋放總線,這樣 IO 口會被上拉電阻自動拉高。
由于 DS18B20 時序要求非常嚴格,所以在操作時序的時候,為了防止中斷干擾總線時序,先關(guān)閉總中斷。然后第一步,拉低 DS18B20 這個引腳,持續(xù) 500us;第二步,IO釋放總線,延時 60us;第三步,讀取存在脈沖,并且等待存在脈沖結(jié)束。
這里寫圖片描述
1.2、ROM 操作指令
總線上可以掛多個器件,通過不同的器件地址來訪問不同的器件。同樣,1-Wire 總線也可以掛多個器件,但是它只有一條線,如何區(qū)分不同的器件呢?
在每個 DS18B20 內(nèi)部都有一個唯一的 64 位長的序列號,這個序列號值就存在 DS18B20內(nèi)部的 ROM 中。開始的 8 位是產(chǎn)品類型編碼(DS18B20 是 0x10),接著的 48 位是每個器件唯一的序號,最后的 8 位是 CRC 校驗碼。DS18B20 可以引出去很長的線,最長可以到幾十米,測不同位置的溫度。單片機可以通過和DS18B20 之間的通信,獲取每個傳感器所采集到的溫度信息,也可以同時給所有的 DS18B20 發(fā)送一些指令。
Skip ROM(跳過 ROM):0xCC
。當總線上只有一個器件的時候,可以跳過 ROM,不進行 ROM 檢測。
1.3、RAM 存儲器操作指令
RAM 讀取指令,常用的就兩條
Read Scratchpad(讀暫存寄存器):0xBE
DS18B20 的溫度數(shù)據(jù)是 2 個字節(jié),我們讀取數(shù)據(jù)的時候,先讀取到的低位,然后才是高位!
Convert Temperature(啟動溫度轉(zhuǎn)換):0x44
當我們發(fā)送一個啟動溫度轉(zhuǎn)換的指令后,DS18B20 開始進行轉(zhuǎn)換。從轉(zhuǎn)換開始到獲取溫度,DS18B20 是需要時間的,而這個時間長短取決于 DS18B20 的精度。前邊說 DS18B20 最高可以用 12 位來存儲溫度,但是也可以用 11 位,10 位和 9 位一共四種格式。位數(shù)越高,精度越高,9 位模式最低位變化 1 個數(shù)字溫度變化 0.5 度,同時轉(zhuǎn)換速度也要快一些。
這里寫圖片描述
其中寄存器 R1 和 R0 決定了轉(zhuǎn)換的位數(shù),出廠默認值就 11,也就是 12 位表示溫度,最大的轉(zhuǎn)換時間是 750ms。當啟動轉(zhuǎn)換后,至少要再等 750ms 之后才能讀取溫度,否則讀到的溫度有可能是錯誤的值。
1.4、DS18B20 的位讀寫時序
寫相關(guān)
當要給 DS18B20 寫入 0 的時候,單片機直接將引腳拉低,持續(xù)時間大于 60us 小于 120us就可以了。圖上顯示的意思是,單片機先拉低 15us 之后,DS18B20 會在從 15us 到 60us 之間的時間來讀取這一位,DS18B20 最早會在 15us 的時刻讀取,典型值是在 30us 的時刻讀取,最多不會超過 60us,DS18B20 必然讀取完畢,所以持續(xù)時間超過 60us 即可。
當要給 DS18B20 寫入 1 的時候,單片機先將這個引腳拉低,拉低時間大于 1us,然后馬上釋放總線,即拉高引腳,并且持續(xù)時間也要大于 60us。和寫 0 類似的是,DS18B20 會在15us 到 60us 之間來讀取這個 1
這里寫圖片描述
讀相關(guān)
當要讀取 DS18B20 的數(shù)據(jù)的時候,我們的單片機首先要拉低這個引腳,并且至少保持1us 的時間,然后釋放引腳,釋放完畢后要盡快讀取。從拉低這個引腳到讀取引腳狀態(tài),不能超過 15us。大家從圖 16-18 可以看出來,主機采樣時間,也就是 MASTER SAMPLES,是在 15us 之內(nèi)必須完成的。
這里寫圖片描述
常用的帶小數(shù)的數(shù)據(jù)處理方法有兩種,一種是定義成浮點型直接處理,第二種是定義成整型,然后把小數(shù)和整數(shù)部分分離出來,在合適的位置點上小數(shù)點即可。
特別強調(diào):DS18B20 的時序比較嚴格,寫的過程中最好不要有中斷打斷,但是在兩個“位”之間的間隔,是大于 1 小于無窮的,那在這個時間段,我們是可以開中斷來處理其它程序的。
二、動手實驗
貼出DS18B20的底層代碼...
#include "config.h"void Delay(u8 us){ do{ _nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_(); }while(--us);}bit Get18B20Ack(){ bit ack; DS18B20_IO = 0; Delay(250); Delay(250); DS18B20_IO = 1; Delay(60); ack = DS18B20_IO; while(!DS18B20_IO); return ack; }void DS18B20Write(u8 dat){ u8 mask; for(mask=0x01; mask!=0; mask<<=1) { DS18B20_IO = 0; Delay(2); if(dat&mask) { DS18B20_IO = 1; } else { DS18B20_IO = 0; } Delay(60); DS18B20_IO = 1; }}u8 DS18B20Read(){ u8 mask, dat=0; for(mask=0x01; mask!=0; mask<<=1) { DS18B20_IO = 0; Delay(2); DS18B20_IO = 1; Delay(2); if(DS18B20_IO) { dat |= mask; } Delay(60); } return dat; }bit Start18B20(){ bit ack; ack = Get18B20Ack(); if(ack == 0) { DS18B20Write(0xCC); DS18B20Write(0x44); } return ~ack;}bit Get18B20Temp(int *temp){ bit ack; u8 LSB, MSB; ack = Get18B20Ack(); if(ack == 0) { DS18B20Write(0xCC); DS18B20Write(0xBE); LSB = DS18B20Read(); MSB = DS18B20Read(); *temp = ((u16)MSB<<8) + LSB; } return ~ack;}
后記
如果想實現(xiàn),顯示小數(shù)點后×位,也是可以的。 小數(shù)在數(shù)碼管上的處理,兩種思路:
- 統(tǒng)一定義為整型,最后再加小數(shù)點。
- 還有一種是定義為浮點型,然后擴大相應(yīng)的倍數(shù)再加小數(shù)點。(emmm,,,好像一樣。^_^)
這里說一種自己的實現(xiàn):通過datasheet其實是可以知道它的精度是0.0625的,然后我們?nèi)绻肴〉叫?shù)點后兩位的話,其實算出來的整數(shù)后乘以6.25就行。
這里寫圖片描述
這里寫圖片描述
小結(jié):本篇文章主要介紹了單片機學(xué)習(xí)中的一個重要模塊:DS18B20。在該部分學(xué)習(xí)中并沒有什么太難的知識點,反而是一些技巧性的東西比較多,這個多多總結(jié)就可以了。
希望大家多多支持我的原創(chuàng)文章。如有錯誤,請大家及時指正,非常感謝。