標題: 51單片機尋跡避障小車源程序 [打印本頁]

作者: augestclown    時間: 2018-7-16 08:21
標題: 51單片機尋跡避障小車源程序
單片機避障程序源程序如下:
  1. /*
  2. * 作者:趙新
  3. * 功能:實現(xiàn)小車躲避障礙,通過檢測三個方向的距離,選擇最大距離轉(zhuǎn)彎
  4. * 日期:3/14
  5. * 說明:STC89c52RC,12MHz
  6. * 注意:1000ms和100ms待測,完成后刪除此行
  7. *—————————————————管腳說明——————————————
  8. * Trig = P1^0
  9. * Echo = P3^2
  10. * PWM_OUT = P0^4
  11. *————————————————————————————————————————
  12. */
  13. #include "stc89c5x.h"
  14. #include "intrins.h"
  15. #include "Motor.h"
  16. #define X 20   //最短距離參考值 約為12厘米 受溫度影響,會存在10%左右的誤差

  17. sbit Trig = P1^0;//發(fā)送端
  18. sbit Echo = P3^2;//接收端 若用外部中斷0,則此引腳必須是P3.2

  19. sbit PWM_OUT = P0^4;//PWM信號輸出端
  20. u8 counts = 0;      //設置初值
  21. u8 PWM =6;                        //設置初值,任意值也可不設
  22. u8 Flag_Angle = 1;  //0 左45度 1 右45度 在函數(shù)Scan()中調(diào)用
  23. u8 Distance_Middle;
  24. u8 Distance_Temp[2];//0 左45度 1 右45度

  25. void Delay20us();
  26. void Delay100ms();
  27. void Delay1000ms();  //@12.000MHz  用于等待
  28. u8 Compute(u8 th,u8 tl);
  29. void Scan_Around();  //掃描左右
  30. void Scan_Middle();  //掃描正中前進方向距離
  31. void main()
  32. {
  33.   TMOD = 0x11;//設置T0,T1 T0用于電平檢測 T1用于產(chǎn)生舵機需要的PWM信號
  34.   
  35.   TH0 = 0x00;           //轉(zhuǎn)載初值
  36.   TL0 = 0x00;
  37.   ET0 = 1;           //打開定時器中斷
  38.   TF0 = 0;           //失能定時器中斷標志,也可忽略此語句
  39.   TR0 = 0;           //開始時T0關(guān)閉

  40.   TH1 =        0xff;  //產(chǎn)生時基100us定時,用于組成舵機各個角度的信號
  41.   TL1 = 0x9c;
  42.   ET1 = 1;
  43.   TF0 = 0;     //可忽略此語句,51復位此位為0
  44.   TR1 = 0;           //暫時關(guān)閉T1
  45.    
  46.   EX0 = 1;     //開外部中斷   
  47.   IT0 = 1;     //下降沿觸發(fā)中斷

  48.   EA = 1;           //全局中斷開
  49.                                    
  50.   Trig = 0;           //觸發(fā)端拉低
  51.   Echo = 0;
  52.   while(1)
  53.   {
  54.     //Scan_Around();
  55.     Scan_Middle();
  56.         if(Distance_Middle<=X)
  57.         {
  58.            Stop();
  59.            Scan_Around();
  60.            if((Distance_Temp[0]>Distance_Temp[1])&&Distance_Temp[0]>=X)
  61.            {
  62.              Turn_Left(0);
  63.                  Delay100ms();
  64.                  //Delay100ms();
  65.            }else
  66.             if((Distance_Temp[0]<Distance_Temp[1])&&Distance_Temp[1]>=X)
  67.                 {
  68.                   Turn_Right(0);
  69.                   Delay100ms();
  70.                   //Delay100ms();
  71.                 }else
  72.                  if((Distance_Temp[0]==Distance_Temp[1])&&Distance_Temp[0]>=X)
  73.                  {
  74.                    Turn_Right(0);
  75.                    Delay100ms();
  76.                    //Delay100ms();
  77.                  }else
  78.                   {
  79.                     Back();
  80.                         Delay100ms();
  81.                         Delay100ms();
  82.                         Delay100ms();
  83.                         Delay100ms();
  84.                         Delay100ms();
  85.                         Turn_Right(0);
  86.                         Delay100ms();
  87.                         //Delay100ms();
  88.                   }
  89.         }else
  90.          {
  91.            Go();
  92.          }
  93.   }
  94. }

  95. void Timer1(void) interrupt 3 //PWM產(chǎn)生
  96. {
  97.   TH1 = 0xff;
  98.   TL1 = 0x9c;
  99.   if(counts<PWM)
  100.     {
  101.           PWM_OUT = 1;
  102.          
  103.         }
  104.   else
  105.     {
  106.          PWM_OUT = 0;
  107.          if(counts==200)
  108.          {
  109.            counts = 0;
  110.          }
  111.         }
  112.          ++counts;
  113. }

  114. void Timer0(void) interrupt 1 //T0溢出中斷函數(shù),一般來說T0溢出是不可能發(fā)生的,原因是傳感器最大探測距離為4m左右,所用時間不會超過65536us
  115. {
  116.   
  117. }
  118. void INT0_Test(void) interrupt 0  //下降沿到來之后,進入外部中斷函數(shù),停止T0計數(shù),計算并發(fā)送計算值到計算機
  119. {
  120.   TR0 = 0;
  121.   switch(Flag_Angle)
  122.   {
  123.     case 0:Distance_Temp[0]=Compute(TH0,TL0);TH0 = TL0 = 0x00;break;  //左
  124.         case 1:Distance_Temp[1]=Compute(TH0,TL0);TH0 = TL0 = 0x00;break;  //右
  125.     default:break;
  126.   }
  127.                     //為了下一次準確計數(shù),必須清空  
  128. }

  129. void Scan_Around()   //掃描左右
  130. {
  131.    Flag_Angle = 0;
  132.    PWM = 17;//左轉(zhuǎn)45度
  133.    TR1 = 1;
  134.    Delay1000ms();
  135.    TR1 = 0;                                 

  136.    Trig = 1;                //觸發(fā)一次檢測
  137.    Delay20us();
  138.    Trig = 0;
  139.    while(!Echo);        //如果沒有檢測到返回信號,等
  140.    TR0 = 1;                //檢測到高電平,開T0計數(shù),一直計到下降沿到來
  141.    Delay100ms();

  142.    Flag_Angle = 1;
  143.    PWM = 8; //右轉(zhuǎn)45度
  144.    TR1 = 1;
  145.    Delay1000ms();
  146.    TR1 = 0;

  147.    Trig = 1;                //觸發(fā)一次檢測
  148.    Delay20us();
  149.    Trig = 0;
  150.    while(!Echo);        //如果沒有檢測到返回信號,等
  151.    TR0 = 1;                //檢測到高電平,開T0計數(shù),一直計到下降沿到來
  152.    Delay100ms();

  153.    PWM = 12; //測完回到正中
  154.    TR1 = 1;
  155.    Delay1000ms();
  156.    TR1 = 0;
  157. }

  158. void Scan_Middle() //掃描正中前進方向距離
  159. {
  160.   //Stop();
  161.   Flag_Angle = 3;
  162.   Trig = 1;                //觸發(fā)一次檢測
  163.   Delay20us();
  164.   Trig = 0;
  165.   while(!Echo);        //如果沒有檢測到返回信號,等
  166.   TR0 = 1;                //檢測到高電平,開T0計數(shù),一直計到下降沿到來
  167.   Delay100ms();
  168.   Distance_Middle = Compute(TH0,TL0);
  169.   TH0 = TL0 = 0x00;
  170.   //Go();
  171. }

  172. u8 Compute(u8 th,u8 tl)
  173. {
  174.    u16 times = 0x0000;
  175.    times = th;
  176.    times = times<<8;
  177.    times |= tl;

  178.    return (times/58);
  179. }
  180. void Delay20us()                //@12.000MHz  用于產(chǎn)生超聲波觸發(fā)信號
  181. {
  182.         unsigned char i;

  183.         _nop_();
  184.         i = 7;
  185.         while (--i);
  186. }

  187. void Delay100ms()                //@12.000MHz
  188. {
  189.         unsigned char i, j;

  190.         i = 195;
  191.         j = 138;
  192.         do
  193.         {
  194.                 while (--j);
  195.         } while (--i);
  196. }

  197. void Delay1000ms()                //@12.000MHz  用于等待
  198. {

  199.         unsigned char i, j, k;

  200.         _nop_();
  201.         i = 8;
  202.         j = 154;
  203.         k = 122;
  204.         do
  205.         {
  206.                 do
  207.                 {
  208.                         while (--k);
  209.                 } while (--j);
  210.         } while (--i);

  211. }
復制代碼


  1. /*
  2. * 作者:趙新
  3. * 功能:尋跡小車主函數(shù)
  4. * 日期:2015/3/10 進行PWM的修改
  5. * 說明:編譯器會有一些警告,主要是定義的一些函數(shù)在這里沒有用到,可以注釋掉或者寫成條件編譯  2015/9/15
  6. */
  7. #include "reg51.h"
  8. #include "intrins.h"
  9. #include "Motor.h"
  10. #define LEFT  0  //左側(cè)觸到黑線
  11. #define RIGHT 1  //右側(cè)觸到黑線
  12. #define ALL   2         //同時觸到黑線
  13. #define NONE  3         //正常運行沒有觸到黑線

  14. u8 i = 0;
  15. sbit Left = P0^4;
  16. sbit Right = P0^5;
  17. sbit PWM_LEFT  = P1^0;
  18. sbit PWM_RIGHT = P1^1;

  19. u8 JudgeMode();
  20. //void Delay100ms();
  21. void main()
  22. {
  23.   TMOD = 0x01;//定時器0,2工作在模式1 50ms時基
  24.   TH0 = 0xfc;
  25.   TL0 = 0x18;
  26.   ET0 = 1;
  27.   EA  = 1;
  28.   TR0 = 1;
  29.   Go();
  30.   while(1)
  31.   {
  32.      switch(JudgeMode())
  33.          {
  34.             case 0: {
  35.                        Turn_Left(0);
  36.                            //Delay100ms();
  37.                            }break;
  38.             case 1: {
  39.                          Turn_Right(0);
  40.                                  //Delay100ms();
  41.                                  }break;
  42.                 case 2: Stop();       break;
  43.                 case 3: Go();         break;
  44.          }
  45.   }
  46. }

  47. u8 JudgeMode()//用于判斷小車此時狀態(tài)
  48. {
  49.    if(Left==0&&Right==1)//左側(cè)觸到黑線,應向左轉(zhuǎn)
  50.      return LEFT;

  51.    if(Left==1&&Right==0)//右側(cè)觸到黑線,應向右轉(zhuǎn)
  52.      return RIGHT;

  53.    if(Left==0&Right==0)//同時觸到黑線,應進一步判斷是‘T’型還是‘十’型路況
  54.      return ALL;

  55.    if(Left==1&&Right==1)//沒有觸到黑線,可能正常運行,可能小車跑偏了
  56.      return NONE;

  57. }






  58. void Time0(void) interrupt 1  //初定周期200ms,空占比50%
  59. {
  60.      
  61.      TH0 = 0xfc;
  62.      TL0 = 0x18;
  63.             ++i;
  64.          if(i>=2) //>=2 1/4空占比
  65.          {
  66.             PWM_LEFT  =         1;
  67. ……………………

  68. …………限于本文篇幅 余下代碼請從51黑下載附件…………
復制代碼

所有資料51hei提供下載:
循跡避障小車.rar (58.16 KB, 下載次數(shù): 121)



作者: 1115204179    時間: 2018-10-27 23:06
你好,能私發(fā)程序給我嗎,我沒黑幣下載不了,1115204179@qq.com            謝謝
作者: 1115204179    時間: 2018-10-27 23:17
樓主可以私發(fā)給我嗎,1115204179@qq.com  謝謝
作者: 啊哈?    時間: 2018-12-20 20:53
可以私法嗎?1173459156@qq.com
作者: 51dian    時間: 2019-3-17 14:27
非常不錯,感謝分享!
作者: 990722    時間: 2021-6-29 09:12
能用嗎正常使用嗎?
作者: 碎碎平安    時間: 2021-11-12 00:08
請問可以正常使用嗎?




歡迎光臨 (http://www.torrancerestoration.com/bbs/) Powered by Discuz! X3.1