乡下人产国偷v产偷v自拍,国产午夜片在线观看,婷婷成人亚洲综合国产麻豆,久久综合给合久久狠狠狠9

  • <output id="e9wm2"></output>
    <s id="e9wm2"><nobr id="e9wm2"><ins id="e9wm2"></ins></nobr></s>

    • 分享

      51黑:誤人子弟的程序分析

       zsok 2016-01-16


      51黑電子論壇:www.51hei.com/bbs/ 精選文章:

      下面程序是我從別的地方搞來的,最后發(fā)現很多錯誤。為了避免誤人子弟,特地作了一下分析,開頭部分for分析是我寫的,程序中紅色的部分我是加上去的。
      在軟件仿真環(huán)境下,把晶振改成12.0Mhz,C51標簽代碼優(yōu)化設為0,見圖附件,測得for語句的延時為:

      for(time=0;time<1;time++);>

      for(time=0;time<2;time++);>

      for(time=0;time<3;time++);>

      for(time=0;time<4;time++);>

      for(time=0;time<5;time++);>

      for(time=0;time<6;time++);>

      for(time=0;time<7;time++);>

      for(time=0;time<8;time++);>

      for(time=0;time<9;time++);>

      for(time=0;time<10;time++);>

      for(time=0;time<20;time++);>

      for(time=0;time<60ime++);>

      for(time=0;time<70ime++);>

      for(time=0;time<80ime++);>

      for(time=0;time<100;time++);>

      for(time=0;time<200;time++);>

      由上可看出,成等差數列,公差d=8。一般地,如果n>m,an=am+8*(n-m)。




      /*****************************************************

      函數功能:將DS18B20傳感器初始化,讀取應答信號

      出口參數:flag

      ***************************************************/

      bit Init_DS18B20(void)

      {

      bitflag; //儲存DS18B20是否存在的標志,flag=0,表示存在;flag=1,表示不存在

      DQ= 1; //先將數據線拉高

      for(time=0;time<2;time++);>略微延時約6微秒。實際是延時23us

      DQ= 0; //再將數據線從高拉低,要求保持480~960us

      for(time=0;time<200;time++);>略微延時約600微秒,以向DS18B20發(fā)出一持續(xù)480~960us的低電平復位脈沖。際是延時1607us,重大錯誤,都超過960us了。

      DQ= 1; //釋放數據線(將數據線拉高)

      for(time=0;time<10;time++);>延時約30us(釋放總線后需等待15~60usDS18B20輸出存在脈沖)。實際是延時87us,延時多過60us,反而能保證讀到存在脈沖,這點誤撞了。

      flag=DQ; //讓單片機檢測是否輸出了存在脈沖(DQ=0表示存在)

      for(time=0;time<200;time++);>延時足夠長時間,等待存在脈沖輸出完畢

      return(flag); //返回檢測成功標志

      }

      /*****************************************************

      函數功能:從DS18B20讀取一個字節(jié)數據

      出口參數:dat

      ***************************************************/

      unsigned char ReadOneChar(void)

      {

      unsignedchar i=0;

      unsignedchar dat; //儲存讀出的一個字節(jié)數據

      for(i=0;i<>

      {

      DQ=1; //先將數據線拉高

      _nop_(); //等待一個機器周期

      DQ= 0; //單片機從DS18B20讀書據時,將數據線從高拉低即啟動讀時序

      dat>>=1;

      _nop_(); //等待一個機器周期

      DQ= 1; //將數據線'人為'拉高,為單片機檢測DS18B20的輸出電平作準備

      for(time=0;time<2;time++);>延時約6us,使主機在15us內采樣。實際是延時23us,超過了15us。

      if(DQ==1)

      dat|=0x80; //如果讀到的數據是1,則將1存入dat

      else

      dat|=0x00;/如果讀到的數據是0,則將0存入dat,將單片機檢測到的電平信號DQ、r

      for(time=0;time<8;time++);>延時3us,兩個讀時序之間必須有大于1us的恢復期。實際是延時71us。思路根本不對,讀時隙至少延時60us,這里又誤撞對了,把至少延時和讀時隙間隔至少1us都包含進了。

      }

      return(dat); //返回讀出的十進制數據

      }

      /*****************************************************

      函數功能:向DS18B20寫入一個字節(jié)數據

      入口參數:dat

      ***************************************************/

      WriteOneChar(unsigned char dat)

      {

      unsignedchar i=0;

      for(i=0; i<8;>

      {

      DQ=1; // 先將數據線拉高

      _nop_(); //等待一個機器周期

      DQ=0; //將數據線從高拉低時即啟動寫時序

      DQ=dat&0x01; //利用與運算取出要寫的某位二進制數據,并將其送到數據線上等待DS18B20采樣

      for(time=0;time<10;time++);>延時約30usDS18B20在拉低后的約15~60us期間從數據線上采樣。實際是延時87us。

      DQ=1; //釋放數據線

      for(time=0;time<1;time++);>延時3us,兩個寫時序間至少需要1us的恢復期 。實際是延時15us。

      dat>>=1; //dat中的各二進制位數據右移1

      }

      for(time=0;time<4;time++);>稍作延時,給硬件一點反應時間。延時39us。

      }

      /*****************************************************

      函數功能:做好讀溫度的準備

      ***************************************************/

      void ReadyReadTemp(void)

      {

      Init_DS18B20(); //DS18B20初始化

      WriteOneChar(0xCC); // 跳過讀序號列號的操作

      WriteOneChar(0x44); // 啟動溫度轉換

      for(time=0;time<100;time++);>溫度轉換需要一點時間。延時807us。

      Init_DS18B20(); //DS18B20初始化

      WriteOneChar(0xCC);//跳過讀序號列號的操作

      WriteOneChar(0xBE);//讀取溫度寄存器,前兩個分別是溫度的低位和高位

      }

      /*****************************************************

      函數功能:延時若干毫秒

      入口參數:n

      ***************************************************/

      voiddelaynms(unsigned char n)

      {

      unsigned char i;

      for(i=0;i<>

      delay1ms();

      }

      照理說void delaynms(unsigned char n),參數n的取值范圍是0~255,因是是無符號數。但在主函數中卻調用delaynms(1000),明顯錯誤,超出取值范圍。



      • C51代碼優(yōu)化


      2樓回復:
      程序的注釋,不是完全對。
      程序通過返復調整得到正確代碼,這是每一個程序員編程過程,
      在這過程中不可能總是修改注釋,
      這樣就導致注釋的不同步。
      延時函數一般情況下是線性。
      原程序的運行機器周期可能不是1us

      樓主回復:
      樓上,我敢保證我的想法是對的。因為把源程序原封不動寫入芯片,測溫是正常顯示的,晶振是12Mhz。當然引腳也是對應的。

      這要動腦再想想哪程序錯了,為什么還能正常用,的確運氣很好。例如源程序的另一個延時程序又是錯的,其注釋思路肯定不對,但其實際效果還是接近1ms的:
      /*****************************************************
      函數功能:延時1ms
      (3j+2)*i=(3×33+2)×10=1010(微秒),可以認為是1毫秒
      ***************************************************/
      void delay1ms()
      {
      unsigned char i,j;
      for(i=0;i<>
      for(j=0;j<33;j++);>
      }

        本站是提供個人知識管理的網絡存儲空間,所有內容均由用戶發(fā)布,不代表本站觀點。請注意甄別內容中的聯系方式、誘導購買等信息,謹防詐騙。如發(fā)現有害或侵權內容,請點擊一鍵舉報。
        轉藏 分享 獻花(0

        0條評論

        發(fā)表

        請遵守用戶 評論公約

        類似文章 更多