久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 1133|回復: 5
打印 上一主題 下一主題
收起左側

單獨的單片機溫度顯示成功了但與時間和在一起又不行?要怎么改代碼

[復制鏈接]
跳轉到指定樓層
樓主
ID:1133439 發表于 2024-10-17 23:20 來自手機 | 只看該作者 |只看大圖 回帖獎勵 |倒序瀏覽 |閱讀模式
大佬們,要怎么改代碼才能讓我的溫度顯示值。單獨的溫度顯示成功了但與時間和在一起又不行


單片機源程序如下:
  1. #include<reg51.h>
  2. #include <intrins.h>
  3. #define uint unsigned int
  4. #define uchar unsigned char
  5. #define yh 0x80 //LCD第一行的初始位置地址
  6. #define er 0xC0 //LCD第二行的初始位置地址
  7. //液晶屏的與C51之間的引腳連接定義(顯示數據線接C51的P0口)
  8. sbit rs = P2^0;
  9. sbit rw = P2^1; //如果硬件上rw接地,就不用寫這句和后面的rw=0了
  10. sbit en = P2^2;
  11. //DS18B20單總線接口與C51之間的引腳連接定義
  12. sbit DQ = P2^3;
  13. //DS1302時鐘芯片與C51之間的引腳連接定義
  14. sbit RST =P2^4;
  15. sbit SCLK=P2^5;
  16. sbit IO  =P2^6;
  17. //校時按鍵與C51的引腳連接定義
  18. sbit key1=P3^5;  //設置鍵
  19. sbit key2=P3^6;  //加鍵
  20. sbit key3=P3^7;  //減鍵
  21. //ACC累加器
  22. sbit ACC0=ACC^0;
  23. sbit ACC7=ACC^7;
  24. //年、月、日 漢字代碼數組
  25. unsigned char character0[24] = {
  26. 0x08,0x0f,0x12,0x0f,0x0a,0x1f,0x02,0x02,  //"年"地址代碼 0x00
  27. 0x0F,0x09,0x0F,0x09,0x0F,0x09,0x13,0x00,  //"月"地址代碼 0x01
  28. 0x0F,0x09,0x09,0x0F,0x09,0x09,0x0F,0x00}; //"日"地址代碼 0x02
  29. //年、月、日顯示的固定字符
  30. uchar code tab1[]={'2','0','-','-',0x00,'-','-',0x01,'-','-',0x02};
  31. //時間、溫度顯示的固定字符
  32. uchar code tab2[]={':','-','-',':','-','-',' ',' ','-','-',0xdf,0x43};
  33. //星期字符:仿真DS1302的周寄存器正常運行計數1~7,表示周日~六
  34. uchar code tab3[8][3]={"***","SUN","MON","TUE","WED","THU","FRI","SAT"};
  35. //緩存 存儲順序是秒分時日月周年
  36. char  data TIME[7]={0,0,12,12,1,05,23};
  37. //DS1302讀出地址,存儲順序是秒分時日月周年
  38. uchar code READ_RTC_ADDR[7]=  {0x81, 0x83, 0x85, 0x87, 0x89, 0x8b, 0x8d};
  39. //DS1302寫入地址,存儲順序是秒分時日月周年
  40. uchar code WRITE_RTC_ADDR[7]= {0x80, 0x82, 0x84, 0x86, 0x88, 0x8a, 0x8c};

  41. uchar Key_value;
  42. bit flag=0;//5毫秒中斷標志
  43. bit mark=1;

  44. //延時函數
  45. void delay(uint xms)//延時函數,有參函數
  46. {
  47.         uint x,y;
  48.         for(x=xms;x>0;x--)
  49.                 for(y=110;y>0;y--);
  50. }
  51. /***********LCD1602判忙函數**********/
  52. void Read_Busy()   //忙檢測函數,判斷bit7是 0執行 1禁止
  53. {
  54.         uchar sta;      //
  55.         rs = 0;
  56.         rw = 1;
  57.         P0 = 0xff;
  58.         do
  59.         {
  60.                 en = 1;
  61.                 sta = P0;
  62.                 en = 0;    //使能
  63.         }while(sta & 0x80);
  64. }
  65. //****液晶寫入指令函數****
  66. void write_1602com(uchar com)
  67. {
  68.         Read_Busy();
  69.         rs=0;//數據/指令選擇置為指令
  70.         rw=0; //讀寫選擇置為寫
  71.         en=1;//拉高使能端,為制造有效的下降沿做準備
  72.         P0=com;//送入數據
  73. //        delay(1);
  74.         en=0;//en由高變低,產生下降沿,液晶執行命令
  75. }
  76. //***液晶寫入數據函數****
  77. void write_1602dat(uchar dat)
  78. {
  79.         Read_Busy();
  80.         rs=1;//數據/指令選擇置為數據
  81.         rw=0; //讀寫選擇置為寫
  82.         en=1; //en置高電平,為制造下降沿做準備
  83.         P0=dat;//送入數據
  84. //        delay(1);
  85.         en=0; //en由高變低,產生下降沿,液晶執行命令
  86. }
  87. //***液晶初始化函數****
  88. void LCD_Init()
  89. {
  90.         uchar i;
  91.         write_1602com(0x38);//設置液晶工作模式
  92.         write_1602com(0x0c);//開顯示不顯示光標
  93.         write_1602com(0x06);//整屏不移動,光標自動右移
  94.         write_1602com(0x01);//清顯示

  95.         write_1602com(yh);//日歷顯示固定符號從第一行第0個位置之后開始顯示
  96.         for(i=0;i<11;i++)
  97.         {
  98.                 write_1602dat(tab1[i]);//向液晶屏寫日歷顯示的固定符號部分
  99.         }
  100.         write_1602com(er+4);//時間顯示固定符號寫入位置,從第2個位置后開始顯示
  101.         for(i=0;i<14;i++)
  102.         {
  103.                 write_1602dat(tab2[i]);//寫顯示時間固定符號,兩個冒號和C
  104.         }
  105. }
  106. /***************DS1302有關子函數********************/
  107. void write_byte(uchar dat)//寫一個字節
  108. {
  109.         uchar i;
  110.         ACC=dat;
  111.         RST=1;
  112.         for(i=8;i>0;i--)
  113.         {
  114.                 IO=ACC0;
  115.                 SCLK=0;
  116.                 SCLK=1;
  117.                 ACC>>=1;
  118.         }
  119. }
  120. uchar read_byte()//讀一個字節
  121. {
  122.         uchar i;
  123.         RST=1;
  124.         for(i=8;i>0;i--)
  125.         {
  126.                 ACC7=IO;
  127.                 SCLK=1;
  128.                 SCLK=0;
  129.                 ACC>>=1;
  130.         }
  131.         return ACC;
  132. }
  133. //----------------------------------------
  134. void write_1302(uchar add,uchar dat)//向1302芯片寫函數,指定寫入地址,數據
  135. {
  136.         RST=0;
  137.         SCLK=0;
  138.         RST=1;
  139.         write_byte(add);
  140.         write_byte(dat);
  141.         SCLK=1;
  142.         RST=0;
  143. }
  144. uchar read_1302(uchar add)//從1302讀數據函數,指定讀取數據來源地址
  145. {
  146.         uchar temp;
  147.         RST=0;
  148.         SCLK=0;
  149.         RST=1;
  150.         write_byte(add);
  151.         temp=read_byte();
  152.         SCLK=1;
  153.         RST=0;
  154.         return(temp);
  155. }
  156. //BCD碼轉十進制函數,輸入BCD碼返回十進制
  157. uchar BCD_D(uchar bcd)//BCD碼轉十進制函數,
  158. {
  159.         return ((bcd>>4)*10)+(bcd & 0x0f);
  160. }
  161. //十進制轉BCD碼函數,輸入十進制返回BCD碼
  162. uchar D_BCD(uchar Dec)
  163. {
  164.         return (Dec/10*16+Dec%10);
  165. }
  166. //--------------------------------------
  167. /*
  168. //1302芯片初始化函數
  169. void ds1302_init()
  170. {
  171.         uchar i,
  172.         RST=0;
  173.         SCLK=0;
  174.         write_1302(0x8e,0x00); //允許寫,禁止寫保護
  175.         for(i=0;i<7;i++)
  176.                 write_1302(WRITE_RTC_ADDR[i],D_BCD(TIME[i])); //向DS1302內寫入初始數據
  177.         write_1302(0x8e,0x80); //打開寫保護
  178. }*/
  179. //刷新LCD函數
  180. write_LCD(uchar add,uchar dat)
  181. {
  182.         write_1602com(add);
  183.         write_1602dat(dat/10+'0');
  184.         write_1602dat(dat%10+'0');
  185. }
  186. //刷新星期函數
  187. void write_WEE(uchar add, uchar dat)
  188. {
  189.         uchar i;
  190.         write_1602com(add);//星期字符的顯示位置
  191.         for(i=0;i<3;i++)
  192.                 write_1602dat(tab3[dat][i]);
  193. }

  194. //按鍵掃描函數
  195. void keyscan()
  196. {
  197.         static bit key_sign=0;//按鍵狀態標志
  198.         static uchar count=0;
  199.         uchar i;
  200.         if(!key1||!key2||!key3)
  201.         {
  202.                 if(++count>=3 && key_sign==0)
  203.                 {
  204.                         key_sign=1;
  205.                         if(!key1)
  206.                         {
  207.                                 Key_value=++Key_value%8;

  208.                                 switch(Key_value)
  209.                                 {       
  210.                                         case 0: write_1602com(0x0c);//按動到第8次,設置光標不閃爍                                               
  211.                                                           write_1302(0x8e,0x00);//允許寫,禁止寫保護
  212.                                                                 for(i=0;i<7;i++)
  213.                                                                         write_1302(WRITE_RTC_ADDR[i],D_BCD(TIME[i])); //向DS1302內寫入數據
  214.                                                           write_1302(0x8e,0x80);//禁止寫,打開寫保護
  215.                                                           mark=1;break;//允許讀1302
  216.                                         case 1: mark=0;//禁止讀DS1302
  217.                                                         write_1602com(0x0f); //設置光標為閃爍
  218.                                                         write_1602com(yh+3); break;//按動第1次,年位置顯示光標
  219.                                         case 2: write_1602com(yh+6); break;//按動第2次,月位置顯示光標               
  220.                                         case 3: write_1602com(yh+9); break;//按動第3次,日位置顯示光標
  221.                                         case 4: write_1602com(yh+14);break;//按動第4次,周位置顯示光標
  222.                                         case 5: write_1602com(er+3); break;//按動第5次,時位置顯示光標
  223.                                         case 6: write_1602com(er+6); break;//按動第6次,分位置顯示光標
  224.                                         case 7: write_1602com(er+9); break;//按動第7次,秒位置顯示光標
  225.                                 }
  226.                         }
  227.                         if(!key2 && Key_value!=0)
  228.                         {
  229.                                 switch(Key_value)
  230.                                 {
  231.                                         case 1:if(++TIME[6]>99)TIME[6]=0; write_LCD(yh+2, TIME[6]);write_1602com(yh+3); break;//調年
  232.                                         case 2:if(++TIME[4]>12)TIME[4]=1; write_LCD(yh+5, TIME[4]);write_1602com(yh+6); break;//調月
  233.                                         case 3:if(++TIME[3]>31)TIME[3]=1; write_LCD(yh+8, TIME[3]);write_1602com(yh+9); break;//調日
  234.                                         case 4:if(++TIME[5]>7) TIME[5]=1; write_WEE(yh+13,TIME[5]);write_1602com(yh+14);break;//調周
  235.                                         case 5:if(++TIME[2]>23)TIME[2]=0; write_LCD(er+2, TIME[2]);write_1602com(er+3); break;//調時
  236.                                         case 6:if(++TIME[1]>59)TIME[1]=0; write_LCD(er+5, TIME[1]);write_1602com(er+6); break;//調分
  237.                                         case 7:if(++TIME[0]>59)TIME[0]=0; write_LCD(er+8, TIME[0]);write_1602com(er+9); break;//調秒
  238.                                 }
  239.                         }
  240.                         if(!key3 && Key_value!=0)
  241.                         {
  242.                                 switch(Key_value)
  243.                                 {
  244.                                         case 1:if(--TIME[6]<0)TIME[6]=99; write_LCD(yh+2, TIME[6]);write_1602com(yh+3); break;//調年
  245.                                         case 2:if(--TIME[4]<1)TIME[4]=12; write_LCD(yh+5, TIME[4]);write_1602com(yh+6); break;//調月
  246.                                         case 3:if(--TIME[3]<1)TIME[3]=31; write_LCD(yh+8, TIME[3]);write_1602com(yh+9); break;//調日
  247.                                         case 4:if(--TIME[5]<1)TIME[5]=7;  write_WEE(yh+13,TIME[5]);write_1602com(yh+14);break;//調周
  248.                                         case 5:if(--TIME[2]<0)TIME[2]=23; write_LCD(er+2, TIME[2]);write_1602com(er+3); break;//調時
  249.                                         case 6:if(--TIME[1]<0)TIME[1]=59; write_LCD(er+5, TIME[1]);write_1602com(er+6); break;//調分
  250.                                         case 7:if(--TIME[0]<0)TIME[0]=59; write_LCD(er+8, TIME[0]);write_1602com(er+9); break;//調秒
  251.                                 }
  252.                         }
  253.                 }
  254.         }
  255.         else
  256.         {
  257.                 key_sign=0;
  258.                 count=0;
  259.         }
  260. }
  261. /*******************************  延時函數 ********************************
  262. *  功能:在11.059MHz的晶振條件下調用本函數需要24μs ,然后每次計數需16μs
  263. **************************************************************************/
  264. void DS18_delay(int us)
  265. {
  266.         int s;
  267.         for (s=0;s<us;s++);
  268. }
  269. /*******************************  復位函數 *******************************
  270. * 功能:完成單總線的復位操作。
  271. * 復位時間為480μs,因此延時時間為(480-24)/16 = 28.5,取29μs。
  272. * 經過70μs之后檢測存在脈沖,因此延時時間為(70-24)/16 = 2.875,取3μs
  273. **************************************************************************/
  274. unsigned char ow_reset(void)
  275. {
  276.         unsigned char presence;
  277.         DQ = 0;                          // 將 DQ 線拉低
  278.         DS18_delay(29);         // 保持 480μs
  279.         DQ = 1;                         // DQ返回高電平
  280.         DS18_delay(3);                         // 等待存在脈沖
  281.         presence = DQ;                 // 獲得存在信號
  282.         DS18_delay(25);                         // 等待時間隙結束
  283.         return(presence);         // 返回存在信號,0 = 器件存在, 1 = 無器件
  284. }
  285. /****************************** 位寫入函數 *******************************
  286. * 功能:向單總線寫入1位值:bitval
  287. *************************************************************************/
  288. void write_bit(char bitval)
  289. {
  290.         DQ = 0;                                 // 將DQ 拉低開始寫時間隙
  291.         if(bitval==1)
  292.                 DQ =1;         // 如果寫1,DQ 返回高電平
  293.         DS18_delay(5);                                 // 在時間隙內保持電平值,
  294.         DQ = 1;         // DS18_delay函數每次循環延時16μs,因此DS18_delay(5) = 104μs
  295. }       
  296. /**************************** 字節寫入函數 *******************************
  297. * 功能:向單總線寫入一個字節值:val
  298. *************************************************************************/
  299. void ds18write_byte(char val)
  300. {
  301.         uchar i;
  302.         uchar temp;
  303.         for (i=0; i<8; i++)
  304.         {        // 寫入字節, 每次寫入一位
  305.                 temp = val>>i;                
  306.                 temp &= 0x01;                
  307.                 write_bit(temp);
  308.         }
  309.         DS18_delay(5);
  310. }
  311. /**************************** 位讀取函數 ********************************
  312. * 功能:從單總線上讀取一位信號,所需延時時間為15μs,因此無法調用前面定義
  313. * 的DS18_delay()函數,而采用一個for()循環來實現延時。
  314. * ***********************************************************************/
  315. uchar read_bit(void)
  316. {
  317.         uchar i;
  318.         DQ = 0;                      //將DQ 拉低開始讀時間隙
  319.         DQ = 1;                         // then return high
  320.         for (i=0; i<3; i++);         // 延時15μs
  321.         return(DQ);                         // 返回 DQ 線上的電平值
  322. }

  323. /**************************** 字節讀取函數 *******************************
  324. * 功能:從單總線讀取一個字節的值
  325. *************************************************************************/
  326. uchar DSread_byte(void)
  327. {
  328.         unsigned char i;
  329.         unsigned char value = 0;
  330.         for (i=0;i<8;i++)
  331.         {                                  // 讀取字節,每次讀取一個字節
  332.                 if(read_bit())
  333.                         value|=0x01<<i;         // 然后將其左移
  334.                 DS18_delay(6);                                        
  335.         }
  336.         return(value);
  337. }
  338. /******************************* 讀取溫度函數 *****************************
  339. * 功能:如果單總線節點上只有一個器件則可以直接調用本函數。如果節點上有多個器
  340. *      件,為了避免數據沖突,應使用Match ROM函數來選中特定器件。
  341. * 注: 本函數是根據DS1820的溫度數據格式編寫的,若用于DS18B20,必須根據
  342. *      DS18B20的溫度數據格式作適當修改。
  343. **************************************************************************/
  344. uint ReadTemperature(void)
  345. {
  346.         unsigned char temp_H,temp_L,temp_T;
  347.         ow_reset();
  348.         ds18write_byte(0xCC);                 // 跳過 ROM
  349.         ds18write_byte(0x44);                 // 啟動溫度轉換
  350.         delay(2);
  351.         ow_reset();
  352.         ds18write_byte(0xCC);                 // 跳過 ROM
  353.         ds18write_byte(0xBE);                 // 讀暫存器
  354.         temp_L=DSread_byte();
  355.         temp_H=DSread_byte();
  356.         temp_T=temp_H<<4|temp_L>>4;//取整數
  357.         return temp_T;
  358.         //temp_f = (((int)temp_c)* 9)/5 + 32;
  359.         //return temp_f; 輸出華氏溫度值
  360. }
  361. //-------------------------------
  362. void Timer0Init()                //5毫秒@11.0592MHz
  363. {
  364.         TMOD |= 0x01;                //設置定時器模式
  365.         TL0 = 0x00;                        //設置定時初始值
  366.         TH0 = 0xEE;                        //設置定時初始值
  367.         TF0 = 0;                                //清除TF0標志
  368.         TR0 = 1;                                //定時器0開始計時
  369.         EA  = 1;                          //開總中斷
  370.         ET0 = 1;                         //開T0中斷
  371. }
  372. //*******************主函數**************************
  373. void main()
  374. {
  375.         uchar h;

  376.         LCD_Init();      //LCD初始化函數
  377. //        ds1302_init();   //DS1302初始化函數
  378.         write_1602com(0x40);//寫1602寄存器命令
  379.         for (h=0;h<24;h++)
  380.         {//向LCD自定義存儲器寫入"年"、"月"、"日"
  381.                 write_1602dat(character0[h]);
  382.         }
  383.         Timer0Init();    //定時器初始化函數
  384.         while(1)
  385.         {                       
  386.                 if(flag)//5毫秒
  387.                 {
  388.                         flag=0;
  389.                         keyscan();      //按鍵掃描函數       
  390.                 }
  391.         }
  392. }

  393. void Timer0Interrupt() interrupt 1
  394. {       
  395.         static uchar i;
  396.         uchar temp;       
  397.         TL0 = 0x00;                        //設置定時初始值
  398.         TH0 = 0xEE;                        //設置定時初始值
  399.         if(mark)
  400.         {
  401.                 switch(i)
  402.                 {
  403.                         case  0: temp=ReadTemperature();       break;//讀溫度
  404.                         case  1: TIME[0] = BCD_D(read_1302(0x81)); break;//讀秒
  405.                         case  2: TIME[1] = BCD_D(read_1302(0x83)); break;//讀分
  406.                         case  3: TIME[2] = BCD_D(read_1302(0x85)); break;//讀時
  407.                         case  4: TIME[3] = BCD_D(read_1302(0x87)); break;//讀日
  408.                         case  5: TIME[4] = BCD_D(read_1302(0x89)); break;//讀月
  409.                         case  6: TIME[5] = BCD_D(read_1302(0x8b)); break;//讀周
  410.                         case  7: TIME[6] = BCD_D(read_1302(0x8d)); break;//讀年
  411.                         case  8: write_LCD(er+12,temp);     break;//刷新溫度
  412.                         case  9: write_LCD(er+8, TIME[0]);    break;//刷新秒
  413.                         case 10: write_LCD(er+5, TIME[1]);    break;//刷新分
  414.                         case 11: write_LCD(er+2, TIME[2]);    break;//刷新時
  415.                         case 12: write_LCD(yh+8, TIME[3]);    break;//刷新日
  416.                         case 13: write_LCD(yh+5, TIME[4]);    break;//刷新月
  417.                         case 14: write_WEE(yh+13,TIME[5]);     break;//刷新周
  418.                         case 15: write_LCD(yh+2, TIME[6]);    break;//刷新年
  419.                         default: break;
  420.                 }       
  421.                 i=++i%16;
  422.         }
  423.         flag=1;
  424. }


復制代碼


分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏 分享淘帖 頂 踩
回復

使用道具 舉報

沙發
ID:1133081 發表于 2024-10-18 05:16 | 只看該作者

萬年歷1602 1302 1820 仿真程序.rar (140.83 KB, 下載次數: 5)

仿真中的電源及晶振都是默認存在,無需添加。實物中需要補齊。
回復

使用道具 舉報

板凳
ID:1121801 發表于 2024-10-18 08:11 | 只看該作者
把數據壓棧沒有?是不是程序運行過程中溫度數據清零了?
回復

使用道具 舉報

地板
ID:229502 發表于 2024-10-18 12:22 | 只看該作者
發代碼,粘貼到網頁上,才好幫你分析
回復

使用道具 舉報

5#
ID:59202 發表于 2024-10-18 18:03 | 只看該作者
那時間能否單獨顯示呢?如果溫度和時間都能單獨顯示,又怎么不能一起顯示呢
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 日本a v在线播放 | 色综合视频在线 | 欧美日韩在线观看视频 | www.色五月.com| 日韩欧美在线视频 | 欧美黄色片 | 97精品一区二区 | 久久久影院 | 亚洲不卡在线观看 | 中文字幕av免费 | 在线播放国产一区二区三区 | 精品国产一区二区在线 | 国产精品theporn | 成人欧美一区二区三区黑人孕妇 | 日韩在线一区二区三区 | 免费黄网站在线观看 | 黄网免费看 | 狠狠干天天干 | 成人久久久 | 亚洲免费在线 | 精品欧美乱码久久久久久 | 三级视频久久 | 国产精品免费一区二区三区四区 | www.五月天婷婷.com | 91精品国产高清久久久久久久久 | 日日骚网 | 午夜视频大全 | 欧美一级二级三级视频 | 国产精品自产拍 | 欧美4p| 狠狠综合久久av一区二区老牛 | 91成人在线视频 | 91在线资源 | 成人精品视频在线观看 | 国产日韩av一区二区 | 日韩一级一区 | 久久久青草婷婷精品综合日韩 | 久久久影院 | 日韩精品一区二区三区中文在线 | 国产精品美女久久久久aⅴ国产馆 | 狠狠操婷婷 |