找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 8599|回复: 8
收起左侧

51单片机寻迹避障小车源程序

  [复制链接]
ID:372259 发表于 2018-7-16 08:21 | 显示全部楼层 |阅读模式
单片机避障程序源程序如下:
  1. /*
  2. * 作者:赵新
  3. * 功能:实现小车躲避障碍,通过检测三个方向的距离,选择最大距离转弯
  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;//发送端
  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度 在函数Scan()中调用
  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用于产生舵机需要的PWM信号
  34.   
  35.   TH0 = 0x00;           //转载初值
  36.   TL0 = 0x00;
  37.   ET0 = 1;           //打开定时器中断
  38.   TF0 = 0;           //失能定时器中断标志,也可忽略此语句
  39.   TR0 = 0;           //开始时T0关闭

  40.   TH1 =        0xff;  //产生时基100us定时,用于组成舵机各个角度的信号
  41.   TL1 = 0x9c;
  42.   ET1 = 1;
  43.   TF0 = 0;     //可忽略此语句,51复位此位为0
  44.   TR1 = 0;           //暂时关闭T1
  45.    
  46.   EX0 = 1;     //开外部中断   
  47.   IT0 = 1;     //下降沿触发中断

  48.   EA = 1;           //全局中断开
  49.                                    
  50.   Trig = 0;           //触发端拉低
  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产生
  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溢出中断函数,一般来说T0溢出是不可能发生的,原因是传感器最大探测距离为4m左右,所用时间不会超过65536us
  115. {
  116.   
  117. }
  118. void INT0_Test(void) interrupt 0  //下降沿到来之后,进入外部中断函数,停止T0计数,计算并发送计算值到计算机
  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.                     //为了下一次准确计数,必须清空  
  128. }

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

  136.    Trig = 1;                //触发一次检测
  137.    Delay20us();
  138.    Trig = 0;
  139.    while(!Echo);        //如果没有检测到返回信号,等
  140.    TR0 = 1;                //检测到高电平,开T0计数,一直计到下降沿到来
  141.    Delay100ms();

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

  147.    Trig = 1;                //触发一次检测
  148.    Delay20us();
  149.    Trig = 0;
  150.    while(!Echo);        //如果没有检测到返回信号,等
  151.    TR0 = 1;                //检测到高电平,开T0计数,一直计到下降沿到来
  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;                //触发一次检测
  163.   Delay20us();
  164.   Trig = 0;
  165.   while(!Echo);        //如果没有检测到返回信号,等
  166.   TR0 = 1;                //检测到高电平,开T0计数,一直计到下降沿到来
  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  用于产生超声波触发信号
  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. * 功能:寻迹小车主函数
  4. * 日期:2015/3/10 进行PWM的修改
  5. * 说明:编译器会有一些警告,主要是定义的一些函数在这里没有用到,可以注释掉或者写成条件编译  2015/9/15
  6. */
  7. #include "reg51.h"
  8. #include "intrins.h"
  9. #include "Motor.h"
  10. #define LEFT  0  //左侧触到黑线
  11. #define RIGHT 1  //右侧触到黑线
  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()//用于判断小车此时状态
  48. {
  49.    if(Left==0&&Right==1)//左侧触到黑线,应向左转
  50.      return LEFT;

  51.    if(Left==1&&Right==0)//右侧触到黑线,应向右转
  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, 下载次数: 119)
回复

使用道具 举报

ID:416370 发表于 2018-10-27 23:06 | 显示全部楼层
你好,能私发程序给我吗,我没黑币下载不了,1115204179@qq.com            谢谢
回复

使用道具 举报

ID:416370 发表于 2018-10-27 23:17 | 显示全部楼层
楼主可以私发给我吗,1115204179@qq.com  谢谢
回复

使用道具 举报

ID:449181 发表于 2018-12-20 20:53 | 显示全部楼层
可以私法吗?1173459156@qq.com
回复

使用道具 举报

ID:492360 发表于 2019-3-17 14:27 | 显示全部楼层
非常不错,感谢分享!
回复

使用道具 举报

ID:945628 发表于 2021-6-29 09:12 | 显示全部楼层
能用吗正常使用吗?
回复

使用道具 举报

ID:974680 发表于 2021-11-12 00:08 来自手机 | 显示全部楼层
请问可以正常使用吗?
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表