本帖最后由 codenew 于 2014-10-6 22:42 編輯
下面程序是我從別的地方搞來(lái)的,最后發(fā)現(xiàn)很多錯(cuò)誤。為了避免誤人子弟,特地作了一下分析,開(kāi)頭部分for分析是我寫(xiě)的,程序中紅色的部分我是加上去的。
在軟件仿真環(huán)境下,把晶振改成12.0Mhz,C51標(biāo)簽代碼優(yōu)化設(shè)為0,見(jiàn)圖附件,測(cè)得for語(yǔ)句的延時(shí)為:
for(time=0;time<1;time++); //15us for(time=0;time<2;time++); //23us for(time=0;time<3;time++); //31us for(time=0;time<4;time++); //39us for(time=0;time<5;time++); //47us for(time=0;time<6;time++); //55us for(time=0;time<7;time++); //63us for(time=0;time<8;time++); //71us for(time=0;time<9;time++); //79us for(time=0;time<10;time++); //87us for(time=0;time<20;time++); //167us for(time=0;time<60ime++); //487us for(time=0;time<70ime++); //567us for(time=0;time<80ime++); //647us for(time=0;time<100;time++); //807us for(time=0;time<200;time++); //1607us 由上可看出,成等差數(shù)列,公差d=8。一般地,如果n>m,an=am+8*(n-m)。
/***************************************************** 函數(shù)功能:將DS18B20傳感器初始化,讀取應(yīng)答信號(hào) 出口參數(shù):flag ***************************************************/ bit Init_DS18B20(void) { bitflag; //儲(chǔ)存DS18B20是否存在的標(biāo)志,flag=0,表示存在;flag=1,表示不存在 DQ= 1; //先將數(shù)據(jù)線(xiàn)拉高 for(time=0;time<2;time++);//略微延時(shí)約6微秒。實(shí)際是延時(shí)23us。 DQ= 0; //再將數(shù)據(jù)線(xiàn)從高拉低,要求保持480~960us for(time=0;time<200;time++);//略微延時(shí)約600微秒,以向DS18B20發(fā)出一持續(xù)480~960us的低電平復(fù)位脈沖。實(shí)際是延時(shí)1607us,重大錯(cuò)誤,都超過(guò)960us了。 DQ= 1; //釋放數(shù)據(jù)線(xiàn)(將數(shù)據(jù)線(xiàn)拉高) for(time=0;time<10;time++);//延時(shí)約30us(釋放總線(xiàn)后需等待15~60us讓DS18B20輸出存在脈沖)。實(shí)際是延時(shí)87us,延時(shí)多過(guò)60us,反而能保證讀到存在脈沖,這點(diǎn)誤撞了。 flag=DQ; //讓單片機(jī)檢測(cè)是否輸出了存在脈沖(DQ=0表示存在) for(time=0;time<200;time++); //延時(shí)足夠長(zhǎng)時(shí)間,等待存在脈沖輸出完畢 return(flag); //返回檢測(cè)成功標(biāo)志 } /***************************************************** 函數(shù)功能:從DS18B20讀取一個(gè)字節(jié)數(shù)據(jù) 出口參數(shù):dat ***************************************************/ unsigned char ReadOneChar(void) { unsignedchar i=0; unsignedchar dat; //儲(chǔ)存讀出的一個(gè)字節(jié)數(shù)據(jù) for(i=0;i<8;i++) { DQ=1; //先將數(shù)據(jù)線(xiàn)拉高 _nop_(); //等待一個(gè)機(jī)器周期 DQ= 0; //單片機(jī)從DS18B20讀書(shū)據(jù)時(shí),將數(shù)據(jù)線(xiàn)從高拉低即啟動(dòng)讀時(shí)序 dat>>=1; _nop_(); //等待一個(gè)機(jī)器周期 DQ= 1; //將數(shù)據(jù)線(xiàn)"人為"拉高,為單片機(jī)檢測(cè)DS18B20的輸出電平作準(zhǔn)備 for(time=0;time<2;time++);//延時(shí)約6us,使主機(jī)在15us內(nèi)采樣。實(shí)際是延時(shí)23us,超過(guò)了15us。 if(DQ==1) dat|=0x80; //如果讀到的數(shù)據(jù)是1,則將1存入dat else dat|=0x00;/如果讀到的數(shù)據(jù)是0,則將0存入dat,將單片機(jī)檢測(cè)到的電平信號(hào)DQ、r for(time=0;time<8;time++);//延時(shí)3us,兩個(gè)讀時(shí)序之間必須有大于1us的恢復(fù)期。實(shí)際是延時(shí)71us。思路根本不對(duì),讀時(shí)隙至少延時(shí)60us,這里又誤撞對(duì)了,把至少延時(shí)和讀時(shí)隙間隔至少1us都包含進(jìn)了。 } return(dat); //返回讀出的十進(jìn)制數(shù)據(jù) } /***************************************************** 函數(shù)功能:向DS18B20寫(xiě)入一個(gè)字節(jié)數(shù)據(jù) 入口參數(shù):dat ***************************************************/ WriteOneChar(unsigned char dat) { unsignedchar i=0; for(i=0; i<8; i++) { DQ=1; // 先將數(shù)據(jù)線(xiàn)拉高 _nop_(); //等待一個(gè)機(jī)器周期 DQ=0; //將數(shù)據(jù)線(xiàn)從高拉低時(shí)即啟動(dòng)寫(xiě)時(shí)序 DQ=dat&0x01; //利用與運(yùn)算取出要寫(xiě)的某位二進(jìn)制數(shù)據(jù),并將其送到數(shù)據(jù)線(xiàn)上等待DS18B20采樣 for(time=0;time<10;time++);//延時(shí)約30us,DS18B20在拉低后的約15~60us期間從數(shù)據(jù)線(xiàn)上采樣。實(shí)際是延時(shí)87us。 DQ=1; //釋放數(shù)據(jù)線(xiàn) for(time=0;time<1;time++);//延時(shí)3us,兩個(gè)寫(xiě)時(shí)序間至少需要1us的恢復(fù)期 。實(shí)際是延時(shí)15us。 dat>>=1; //將dat中的各二進(jìn)制位數(shù)據(jù)右移1位 } for(time=0;time<4;time++);//稍作延時(shí),給硬件一點(diǎn)反應(yīng)時(shí)間。延時(shí)39us。 } /***************************************************** 函數(shù)功能:做好讀溫度的準(zhǔn)備 ***************************************************/ void ReadyReadTemp(void) { Init_DS18B20(); //將DS18B20初始化 WriteOneChar(0xCC); // 跳過(guò)讀序號(hào)列號(hào)的操作 WriteOneChar(0x44); // 啟動(dòng)溫度轉(zhuǎn)換 for(time=0;time<100;time++);//溫度轉(zhuǎn)換需要一點(diǎn)時(shí)間。延時(shí)807us。 Init_DS18B20(); //將DS18B20初始化 WriteOneChar(0xCC);//跳過(guò)讀序號(hào)列號(hào)的操作 WriteOneChar(0xBE);//讀取溫度寄存器,前兩個(gè)分別是溫度的低位和高位 } /***************************************************** 函數(shù)功能:延時(shí)若干毫秒 入口參數(shù):n ***************************************************/ voiddelaynms(unsigned char n) { unsigned char i; for(i=0;i<n;i++) delay1ms(); } 照理說(shuō)void delaynms(unsigned char n),參數(shù)n的取值范圍是0~255,因是是無(wú)符號(hào)數(shù)。但在主函數(shù)中卻調(diào)用delaynms(1000),明顯錯(cuò)誤,超出取值范圍。 |