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

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 2076|回復: 0
收起左側

基于51單片機的超聲波測距+LCD顯示+光敏電阻調光程序調試

[復制鏈接]
ID:1065040 發表于 2023-3-8 23:02 | 顯示全部樓層 |閱讀模式
最近我在做基于51單片機的超聲波測距+LCD顯示+光敏電阻調光,先在B站up且坐聽風的教學視頻中仿真出了單路超聲波測距(帶按鍵報警設置),后面想加上光敏電阻調光,于是在本站找到基于51單片機的智能臺燈帶坐姿矯正Proteus仿真設計  http://www.zg4o1577.cn/bbs/dpj-226910-1.html。將兩者結合在一起,但是我在Proteus上畫完圖之后,再導入程序,結果只有超聲波測距功能可以顯現,光敏電阻調光這個功能不能實現。我在找原因以為光敏電阻用了定時器0與超聲波所用的定時器發生沖突,但我修改超聲波用定時器1后,仍然不能實現功能。我在仿真上面測變阻器上兩端的電壓確實隨著阻值的變化和變化,但是輸出燈的那一個引腳卻不變化。以下是單片機程序和仿真圖

仿真圖

仿真圖

  1. #include <reg52.h>
  2. #include <intrins.h>

  3. #define uchar unsigned char        //宏定義
  4. #define uint  unsigned int        

  5. sfr ISP_DATA  = 0xe2;                        // 數據寄存器
  6. sfr ISP_ADDRH = 0xe3;                        // 地址寄存器高八位
  7. sfr ISP_ADDRL = 0xe4;                        // 地址寄存器低八位
  8. sfr ISP_CMD   = 0xe5;                        // 命令寄存器
  9. sfr ISP_TRIG  = 0xe6;                        // 命令觸發寄存器
  10. sfr ISP_CONTR = 0xe7;                        // 命令寄存器

  11. sbit LcdRs_P   = P1^1;                // 1602液晶的RS管腳      
  12. sbit LcdRw_P   = P1^2;                // 1602液晶的RW管腳
  13. sbit LcdEn_P   = P1^3;                // 1602液晶的EN管腳

  14. sbit Trig4_P   = P1^4;                // 超聲波模塊4的Trig管腳
  15. sbit Echo4_P   = P1^5;                // 超聲波模塊4的Echo管腳
  16. sbit KeySet_P  = P2^2;                // 設置按鍵的管腳
  17. sbit KeyDown_P = P2^1;                // 減按鍵的管腳
  18. sbit KeyUp_P   = P2^0;                // 加按鍵的管腳
  19. sbit Buzzer_P  = P2^3;                // 蜂鳴器的管腳

  20. sbit LED     = P3^0;                                        // 模式指示燈,亮是自動模式,滅是手動模式
  21. sbit Lamp    = P3^4;                                         // 臺燈控制引腳
  22. sbit Key1    = P3^1;                                        // 按鍵1,模式切換按鍵
  23. sbit Key2    = P3^2;                                         // 按鍵2,亮度減少按鍵      
  24. sbit Key3    = P3^3;                                        // 按鍵3,亮度增加按鍵
  25. sbit ADC_CS  = P2^7;                                         // ADC0832的CS引腳
  26. sbit ADC_CLK = P2^4;                                         // ADC0832的CLK引腳
  27. sbit ADC_DAT = P2^5;                                         // ADC0832的DI/DO引腳

  28. sbit Led4_P    = P1^0;                // 傳感器4報警燈

  29. uint gAlarm;                                                        // 報警距離變量
  30. uchar gCount=0;                                                                // 全局計數變量
  31. uchar gIndex;                                                                        // 亮度變量,0是最暗,9是最亮,一共10檔
  32. uint  gTime=0;                                                                // 計時變量,用于計時多久沒檢測到有人



  33. void ISP_Disable()
  34. {
  35.         ISP_CONTR = 0;
  36.         ISP_ADDRH = 0;
  37.         ISP_ADDRL = 0;
  38. }


  39. // 從單片機內部EEPROM讀一個字節,從0x2000地址開始

  40. unsigned char EEPROM_Read(unsigned int add)
  41. {
  42.         ISP_DATA  = 0x00;
  43.         ISP_CONTR = 0x83;
  44.         ISP_CMD   = 0x01;
  45.         ISP_ADDRH = (unsigned char)(add>>8);
  46.         ISP_ADDRL = (unsigned char)(add&0xff);
  47.         // 對STC89C51系列來說,每次要寫入0x46,再寫入0xB9,ISP/IAP才會生效
  48.         ISP_TRIG  = 0x46;           
  49.         ISP_TRIG  = 0xB9;
  50.         _nop_();
  51.         ISP_Disable();
  52.         return (ISP_DATA);
  53. }


  54. // 往單片機內部EEPROM寫一個字節,從0x2000地址開始

  55. void EEPROM_Write(unsigned int add,unsigned char ch)
  56. {
  57.         ISP_CONTR = 0x83;
  58.         ISP_CMD   = 0x02;
  59.         ISP_ADDRH = (unsigned char)(add>>8);
  60.         ISP_ADDRL = (unsigned char)(add&0xff);
  61.         ISP_DATA  = ch;
  62.         ISP_TRIG  = 0x46;
  63.         ISP_TRIG  = 0xB9;
  64.         _nop_();
  65.         ISP_Disable();
  66. }



  67. // 寫8個扇區中隨便一個的地址,便擦除該扇區,寫入前要先擦除

  68. void Sector_Erase(unsigned int add)         
  69. {
  70.         ISP_CONTR = 0x83;
  71.         ISP_CMD   = 0x03;
  72.         ISP_ADDRH = (unsigned char)(add>>8);
  73.         ISP_ADDRL = (unsigned char)(add&0xff);
  74.         ISP_TRIG  = 0x46;
  75.         ISP_TRIG  = 0xB9;
  76.         _nop_();
  77.         ISP_Disable();
  78. }



  79. // 延時函數

  80. void DelayMs(uint time)
  81. {
  82.         uint i,j;
  83.         for(i=0;i<time;i++)
  84.                 for(j=0;j<112;j++);
  85. }


  86. // 1602液晶寫命令函數

  87. void LcdWriteCmd(uchar cmd)
  88. {
  89.         LcdRs_P = 0;
  90.         LcdRw_P = 0;
  91.         LcdEn_P = 0;
  92.         P0=cmd;
  93.         DelayMs(2);
  94.         LcdEn_P = 1;   
  95.         DelayMs(2);
  96.         LcdEn_P = 0;        
  97. }



  98. // 1602液晶寫數據函數,dat就是要寫入的數據

  99. void LcdWriteData(uchar dat)
  100. {
  101.         LcdRs_P = 1;
  102.         LcdRw_P = 0;
  103.         LcdEn_P = 0;
  104.         P0=dat;
  105.         DelayMs(2);
  106.         LcdEn_P = 1;   
  107.         DelayMs(2);
  108.         LcdEn_P = 0;
  109. }


  110. // 液晶光標定位函數

  111. void LcdGotoXY(uchar line,uchar column)
  112. {
  113.         // 第一行
  114.         if(line==0)        
  115.                 LcdWriteCmd(0x80+column);
  116.         // 第二行
  117.         if(line==1)        
  118.                 LcdWriteCmd(0x80+0x40+column);
  119. }




  120. // 液晶輸出字符串函數

  121. void LcdPrintStr(uchar *str)
  122. {
  123.         while(*str!='\0')
  124.                         LcdWriteData(*str++);
  125. }



  126. // 液晶輸出數字

  127. void LcdPrintNum(uint num)
  128. {
  129.         LcdWriteData(num/100+0x30);                                // 百位
  130.         LcdWriteData(num%100/10+0x30);                // 十位
  131.         LcdWriteData(num%10+0x30);                                // 個位
  132. }


  133. // 1602液晶功能初始化

  134. void LcdInit()
  135. {
  136.         LcdWriteCmd(0x38);        // 16*2顯示,5*7點陣,8位數據口
  137.         LcdWriteCmd(0x0C);        // 開顯示,不顯示光標
  138.         LcdWriteCmd(0x06);        // 地址加1,當寫入數據后光標右移
  139.         LcdWriteCmd(0x01);        // 清屏
  140. }




  141. // 1602液晶顯示內容初始化

  142. void LcdShowInit()
  143. {
  144.         LcdGotoXY(0,0);                                            // 定位到第0行第0列
  145.         LcdPrintStr("L:       cm");        // 第0行顯示“      U         ”
  146.         LcdGotoXY(1,0);                                                                    // 定位到第1行第0列

  147. }




  148. // 計算傳感器4測量到的距離

  149. uint GetDistance4(void)
  150. {
  151.         uint ss;                                        // 用于記錄測得的距離

  152.         TH0=0;
  153.         TL0=0;

  154.         Trig4_P=1;                                // 給超聲波模塊4一個開始脈沖
  155.         DelayMs(1);
  156.         Trig4_P=0;

  157.         while(!Echo4_P);        // 等待超聲波模塊4的返回脈沖
  158.         TR0=1;                                                // 啟動定時器,開始計時
  159.         while(Echo4_P);                // 等待超聲波模塊4的返回脈沖結束
  160.         TR0=0;                                                // 停止定時器,停止計時

  161.         ss=((TH0*256+TL0)*0.034)/2+1;                // 距離cm=(時間us * 速度cm/us)/2
  162.         return ss;
  163. }



  164. // 按鍵掃描

  165. void KeyScanf()
  166. {
  167.         if(KeySet_P==0)                                        // 判斷是否有按鍵按下
  168.         {
  169.                 LcdGotoXY(0,0);                                                                    // 光標定位
  170.                 LcdPrintStr("   Alarm Set    ");        // 第0行顯示“   Alarm Set    ”
  171.                 LcdGotoXY(1,0);                                                                                // 光標定位
  172.                 LcdPrintStr("  alarm=   cm   ");        // 第1行顯示“  alarm=   cm   ”
  173.                 LcdGotoXY(1,8);                                                                                // 光標定位
  174.                 LcdPrintNum(gAlarm);                                                        // 顯示當前的報警值

  175.                 DelayMs(10);                                                                                        // 消除按鍵按下的抖動
  176.                 while(!KeySet_P);                                                                        // 等待按鍵釋放
  177.                 DelayMs(10);                                                                                        // 消除按鍵松開的抖動               

  178.                 while(1)
  179.                 {               
  180.                         /* 報警值減的處理 */
  181.                         if(KeyDown_P==0)                                       
  182.                         {
  183.                                 if(gAlarm>2)                                                // 報警值大于2才能減1
  184.                                         gAlarm--;                                                        // 報警值減1
  185.                                 LcdGotoXY(1,8);                                        // 光標定位
  186.                                 LcdPrintNum(gAlarm);                // 刷新修改后的報警值
  187.                                 DelayMs(300);                                                // 延時
  188.                         }

  189.                         /* 報警值加的處理 */
  190.                         if(KeyUp_P==0)                                       
  191.                         {
  192.                                 if(gAlarm<400)                                        // 報警值小于400才能加1
  193.                                         gAlarm++;                                                        // 報警值加1
  194.                                 LcdGotoXY(1,8);                                        // 光標定位
  195.                                 LcdPrintNum(gAlarm);                // 刷新修改后的報警值
  196.                                 DelayMs(300);                                                // 延時
  197.                         }
  198.                         
  199.                         /* 退出報警值設置 */
  200.                         if(KeySet_P==0)                                       
  201.                         {
  202.                                 break;                                                                        // 退出while循環
  203.                         }        
  204.                 }
  205.                
  206.                 LcdShowInit();                                                        // 液晶恢復測量到測量界面
  207.                 DelayMs(10);                                                          // 消除按鍵按下的抖動
  208.                 while(!KeySet_P);                                                // 等待按鍵釋放
  209.                 DelayMs(10);                                                          // 消除按鍵松開的抖動               

  210.                 Sector_Erase(0x2000);                                // 保存報警距離
  211.                 EEPROM_Write(0x2000,gAlarm/100);
  212.                 EEPROM_Write(0x2001,gAlarm%100);               
  213.         }        
  214. }





  215. // 傳感器4報警判斷

  216. void AlarmJudge4(uint ss)
  217. {
  218.         if(ss<gAlarm)                // LED燈判斷
  219.         {
  220.                 Led4_P=0;
  221.         }
  222.         else
  223.         {
  224.                 Led4_P=1;
  225.         }
  226.         
  227.         if(Led4_P==0)        // 蜂鳴器判斷
  228.         {
  229.                 Buzzer_P=0;
  230.         }
  231.         else
  232.         {
  233.                 Buzzer_P=1;
  234.         }
  235. }

  236. // 報警值初始化

  237. void AlarmInit()
  238. {
  239.         gAlarm=EEPROM_Read(0x2000)*100+EEPROM_Read(0x2001);                // 從EEPROM讀取報警值

  240.         if((gAlarm==0)||(gAlarm>400))                        // 如果讀取到的報警值異常
  241.         {
  242.                 gAlarm=25;                                                                                        // 重新賦值報警值為25
  243.         }
  244. }

  245. /*********************************************************/
  246. // ADC0832的時鐘脈沖
  247. /*********************************************************/
  248. void WavePlus()
  249. {
  250.         _nop_();
  251.         ADC_CLK = 1;
  252.         _nop_();
  253.         ADC_CLK = 0;
  254. }



  255. /*********************************************************/
  256. // 獲取指定通道的A/D轉換結果
  257. /*********************************************************/
  258. uchar Get_ADC0832()
  259. {
  260.         uchar i;
  261.         uchar dat1=0;
  262.         uchar dat2=0;
  263.         
  264.         ADC_CLK = 0;                                // 電平初始化
  265.         ADC_DAT = 1;
  266.         _nop_();
  267.         ADC_CS = 0;
  268.         WavePlus();                                        // 起始信號
  269.         ADC_DAT = 1;
  270.         WavePlus();                                        // 通道選擇的第一位
  271.         ADC_DAT = 0;      
  272.         WavePlus();                                        // 通道選擇的第二位
  273.         ADC_DAT = 1;
  274.         
  275.         for(i=0;i<8;i++)                // 第一次讀取
  276.         {
  277.                 dat1<<=1;
  278.                 WavePlus();
  279.                 if(ADC_DAT)
  280.                         dat1=dat1|0x01;
  281.                 else
  282.                         dat1=dat1|0x00;
  283.         }
  284.         
  285.         for(i=0;i<8;i++)                // 第二次讀取
  286.         {
  287.                 dat2>>= 1;
  288.                 if(ADC_DAT)
  289.                         dat2=dat2|0x80;
  290.                 else
  291.                         dat2=dat2|0x00;
  292.                 WavePlus();
  293.         }
  294.         
  295.         _nop_();                                                // 結束此次傳輸
  296.         ADC_DAT = 1;
  297.         ADC_CLK = 1;
  298.         ADC_CS  = 1;   

  299.         if(dat1==dat2)                        // 返回采集結果
  300.                 return dat1;
  301.         else
  302.                 return 0;
  303. }



  304. /*********************************************************/
  305. // 定時器初始化
  306. /*********************************************************/
  307. void TimerInit()
  308. {
  309.         TMOD = 0x01;                                // 使用定時器0,工作方式1         
  310.         TH0  = 252;                                        // 給定時器0的TH0裝初值
  311.         TL0  = 24;                                        // 給定時器0的TL0裝初值        
  312.         ET0  = 1;                                                // 定時器0中斷使能
  313.         EA   = 1;                                                // 打開總中斷
  314.         TR0         = 1;                                                // 啟動定時器0
  315. }



  316. /*********************************************************/
  317. // 手動控制
  318. /*********************************************************/
  319. void ManualControl()
  320. {
  321.         // 亮度減少
  322.         if(Key2==0)                                        // 如果按鍵2被按下去
  323.         {
  324.                 if(gIndex>0)                        // 只要當前亮度不為最低才能減少亮度
  325.                 {
  326.                         gIndex--;                                // 亮度降低一檔
  327.                         DelayMs(300);                // 延時0.3秒
  328.                 }
  329.         }
  330.         
  331.         // 亮度增加
  332.         if(Key3==0)                                        // 如果按鍵3被按下去
  333.         {
  334.                 if(gIndex<9)                        // 只要當前亮度不為最高才能增加亮度
  335.                 {
  336.                         gIndex++;                                // 亮度增加一檔
  337.                         DelayMs(300);                // 延時0.3秒
  338.                 }
  339.         }
  340. }



  341. /*********************************************************/
  342. // 自動控制
  343. /*********************************************************/
  344. void AutoControl(uchar num)
  345. {
  346.         if(num<59)                                                                                                                // 最亮
  347.                 gIndex=9;
  348.         else if((num>65)&&(num<81))                                                // 第二亮
  349.                 gIndex=8;
  350.         else if((num>87)&&(num<103))                                        // 第三亮
  351.                 gIndex=7;
  352.         else if((num>109)&&(num<125))
  353.                 gIndex=6;
  354.         else if((num>131)&&(num<147))
  355.                 gIndex=5;
  356.         else if((num>153)&&(num<169))
  357.                 gIndex=4;
  358.         else if((num>175)&&(num<191))
  359.                 gIndex=3;
  360.         else if((num>197)&&(num<213))
  361.                 gIndex=2;
  362.         else if((num>219)&&(num<235))
  363.                 gIndex=1;
  364.         else if(num>241)                                                                                 // 最暗
  365.                 gIndex=0;
  366. }

  367. //超聲波初始化
  368. void Init_ultrasonic_wave()
  369. {
  370.          Trig4_P=0;                           //關閉發射
  371.          TMOD|=0x10;                   //設T1為方式1,GATE=1;
  372.          TH1=0;
  373.          TL1=0;         
  374.          ET1=1;           //允許T1中斷
  375.          EA=1;                           //開啟總中斷        
  376. }


  377. // 主函數

  378. void main()
  379. {
  380.         uchar i;                                                                                // 循環變量
  381.         uint dist;                                                                // 保存測量結果
  382.         uchar ret;
  383.         
  384.         LcdInit();                                                                // 液晶功能初始化
  385.         LcdShowInit();                                                // 液晶顯示內容初始化
  386.         AlarmInit();                                                        // 報警值初始化
  387.         
  388.         Trig4_P=0;

  389.         
  390.         TimerInit();                                         // 定時器初始化
  391.         Init_ultrasonic_wave();          //超聲波定時器初始化
  392.         LED=0;                                                // 指示燈點亮(自動模式指示燈)
  393.         ret=Get_ADC0832();                // 獲取AD采集結果(環境光照強度)
  394.         AutoControl(ret);                        // 上電先進行一次自動亮度控制        
  395.         AutoControl(ret+7);
  396.         gIndex=9;
  397.         
  398.         

  399.         


  400.         while(1)
  401.         {               
  402.                
  403.                 dist=GetDistance4();                // 讀取超聲波模塊4測量到的距離
  404.                 LcdGotoXY(0,2);                            // 光標定位
  405.                 LcdPrintNum(dist);                        // 顯示傳感器4測量到的距離
  406.                 AlarmJudge4(dist);                        // 判斷傳感器4的測量距離是否需要報警
  407.                
  408.                
  409.         
  410.         
  411.                 for(i=0;i<15;i++)
  412.                 {
  413.                         KeyScanf();
  414.                         DelayMs(10);
  415.                 }
  416.                
  417.                
  418.                
  419.                 /* 模式切換控制 */
  420.                 if(Key1==0)                                        // 如果按鍵1被按下去
  421.                 {
  422.                         LED=~LED;                                        // 切換LED燈狀態
  423.                         DelayMs(10);                        // 延時消除按鍵按下的抖動
  424.                         while(!Key1);                        // 等待按鍵釋放
  425.                         DelayMs(10);                        // 延時消除按鍵松開的抖動
  426.                 }
  427.                         
  428.                 /* 亮度控制 */
  429.                 if(LED==1)                                                        // 如果LED是滅的
  430.                 {
  431.                         ManualControl();                        // 則進行手動控制
  432.                         DelayMs(200);
  433.                 }
  434.                 else                                                                                // 如果LED是亮的
  435.                 {
  436.                         
  437.                                 ret=Get_ADC0832();                // 獲取AD采集結果(環境光照強度)
  438.                                 AutoControl(ret);                        // 進行自動控制        
  439.                                 DelayMs(200);
  440.                         
  441.                 }
  442.                
  443.                 }
  444.         
  445. }



  446. /*********************************************************/
  447. // 定時器0服務程序,1毫秒
  448. /*********************************************************/
  449. void Timer0(void) interrupt 1
  450. {
  451.         TH0  = 252;                                                // 給定時器0的TH0裝初值
  452.         TL0  = 24;                                                // 給定時器0的TL0裝初值        
  453.         
  454.         gTime++;                                                        // 每1毫秒,gTime變量加1
  455.         gCount++;                                                        // 每1毫秒,gCount變量加1
  456.         
  457.         if(gCount==10)                                // 如果gCount加到10了
  458.         {
  459.                 gCount=0;                                                // 則將gCount清零,進入新一輪的計數
  460.                 if(gIndex!=0)                                // 如果說臺燈不是最暗的(熄滅)
  461.                 {
  462.                         Lamp=0;                                                // 則把臺燈點亮
  463.                 }
  464.         }
  465.         if(gCount==gIndex)                // 如果gCount計數到和gIndex一樣了
  466.         {
  467.                 if(gIndex!=9)                                // 如果說臺燈不是最亮的
  468.                 {
  469.                         Lamp=1;                                                // 則把臺燈熄滅
  470.                 }
  471.         }
  472. }


  473. void time1() interrupt 3                  //T1中斷用來計數器溢出,超過測距范圍
  474. {
  475.         TH1=0;
  476.         TL1=0;
  477. }  
復制代碼

回復

使用道具 舉報

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

本版積分規則

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

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 午夜欧美一区二区三区在线播放 | 成人亚洲网 | 高清视频一区二区三区 | 日本国产精品视频 | 久久青草av | 久久精品毛片 | www国产成人免费观看视频,深夜成人网 | 黑人巨大精品欧美一区二区免费 | 91免费高清视频 | 亚洲国产精品久久 | 日本一区二区视频 | 91视视频在线观看入口直接观看 | 免费看欧美一级片 | 黄色一级视频 | 激情欧美一区二区三区中文字幕 | 玖玖免费 | 性高湖久久久久久久久aaaaa | h视频在线免费 | 亚洲成av人影片在线观看 | 男人影音 | 亚洲一区黄色 | 一区精品国产欧美在线 | 国产精品高潮呻吟久久 | 久草新在线 | 一区二区三区欧美 | 3p视频在线观看 | 日日射影院 | av激情影院| 九九热视频这里只有精品 | 蜜桃视频成人 | 视频一区二区三区中文字幕 | 成人网在线观看 | jlzzxxxx18hd护士| 一区视频在线免费观看 | 亚洲精品欧美精品 | 欧美亚洲综合久久 | 天堂va在线 | 亚洲一区二区三区在线视频 | 国产真实精品久久二三区 | 天天干夜夜| 欧美日韩一二三区 |