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