找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 6544|回复: 6
收起左侧

51单片机智能车超声波舵机避障程序分享

  [复制链接]
ID:893682 发表于 2021-4-25 16:40 | 显示全部楼层 |阅读模式
附加程序
  1. #include <reg52.h>//51头文件
  2. #include <QX_A11.h>//QX_A11智能小车配置文件
  3. #include <intrins.h>
  4. bit Timer1Overflow;        //计数器1溢出标志位
  5. unsigned char disbuff[4]={0,0,0,0};//用于分别存放距离的值米,厘米,毫米
  6. unsigned int LeftDistance = 0, RightDistance = 0, FrontDistance = 0; //云台测距距离缓存
  7. unsigned int DistBuf[5] = {0};//distance data buffer
  8. unsigned char        pwm_val_left,pwm_val_right;        //中间变量,用户请勿修改。
  9. unsigned char         pwm_left,pwm_right;                        //定义PWM输出高电平的时间的变量。用户操作PWM的变量。
  10. #define                PWM_DUTY                200                        //(用于舵机时不可修改)定义PWM的周期,数值为定时器0溢出周期,假如定时器溢出时间为100us,则PWM周期为20ms。
  11. #define                PWM_HIGH_MIN        70                        //限制PWM输出的最小占空比。用户请勿修改。
  12. #define                PWM_HIGH_MAX        PWM_DUTY        //限制PWM输出的最大占空比。用户请勿修改。

  13. void Timer0_Init(void); //定时器0初始化
  14. void Timer1_Init(void);//定时器1初始化
  15. void LoadPWM(void);//装入PWM输出值
  16. void Delay_Ms(unsigned int ms);//毫秒级延时函数
  17. void forward(unsigned char LeftSpeed,unsigned char RightSpeed);//QX_A11智能小车前进
  18. void left_run(unsigned char LeftSpeed,unsigned char RightSpeed);//QX_A11智能小车左转  
  19. void right_run(unsigned char LeftSpeed,unsigned char RightSpeed);//QX_A11智能小车右转
  20. //void back_run(unsigned char LeftSpeed,unsigned char RightSpeed);//QX_A11智能小车后退
  21. void stop(void);//QX_A11智能小车停车
  22. void Delay18450us(void);        //@11.0592MHz
  23. void Delay1550us(void);                //@11.0592MHz
  24. void Delay19400us(void);        //@11.0592MHz
  25. void Delay600us(void);                //@11.0592MHz
  26. void Delay17500us(void);        //@11.0592MHz
  27. void Delay2500us(void);                //@11.0592MHz
  28. void QXMBOT_bubble(unsigned int *a,unsigned char n);//冒泡排序
  29. void QXMBOT_ServoFront(void);//舵机向前
  30. void QXMBOT_ServoLeft(void);//舵机左转
  31. void QXMBOT_ServoRight(void);//舵机右转
  32. void  QXMBOT_PTZ_Avoid(unsigned int val);//舵机云台避障
  33. unsigned int QXMBOT_GetDistance(void);//获取超声波距离
  34. void QXMBOT_TrigUltrasonic(void);// 触发超声波
  35. unsigned int QXMBOT_RefreshDistance(void);//超声波测距

  36. /*主函数*/     
  37. void main(void)
  38. {
  39.         Timer0_Init();//定时0初始化
  40.         Timer1_Init();//定时0初始化
  41.         EA_on;        //开中断
  42.         QXMBOT_ServoFront();//舵机向前
  43.         Delay_Ms(1000);
  44.         forward(120,120);//前进
  45.         while(1)
  46.         {
  47.                 QXMBOT_PTZ_Avoid(300);//舵机云台避障,LCD1602显示距离,形参设置触发距离,单位:毫米
  48.         }        
  49. }


  50. /*********************************************
  51. QX_A11智能小车前进
  52. 入口参数:LeftSpeed,RightSpeed
  53. 出口参数: 无
  54. 说明:LeftSpeed,RightSpeed分别设置左右车轮转速
  55. **********************************************/
  56. void forward(unsigned char LeftSpeed,unsigned char RightSpeed)
  57. {
  58.         pwm_left = LeftSpeed,pwm_right =  RightSpeed;//设置速度
  59.         left_motor_go; //左电机前进
  60.         right_motor_go; //右电机前进
  61. }
  62. /*小车左转*/
  63. /*********************************************
  64. QX_A11智能小车左转
  65. 入口参数:LeftSpeed,RightSpeed
  66. 出口参数: 无
  67. 说明:LeftSpeed,RightSpeed分别设置左右车轮转速
  68. **********************************************/
  69. void left_run(unsigned char LeftSpeed,unsigned char RightSpeed)
  70. {
  71.         pwm_left = LeftSpeed,pwm_right =  RightSpeed;//设置速度
  72.         left_motor_back; //左电机后退
  73.         right_motor_go; //右电机前进        
  74. }

  75. /*********************************************
  76. QX_A11智能小车右转
  77. 入口参数:LeftSpeed,RightSpeed
  78. 出口参数: 无
  79. 说明:LeftSpeed,RightSpeed分别设置左右车轮转速
  80. **********************************************/
  81. void right_run(unsigned char LeftSpeed,unsigned char RightSpeed)
  82. {
  83.         pwm_left = LeftSpeed,pwm_right =  RightSpeed;//设置速度
  84.         right_motor_back;//右电机后退
  85.         left_motor_go;    //左电机前进
  86. }
  87. /*********************************************
  88. QX_A11智能小车后退
  89. 入口参数:LeftSpeed,RightSpeed
  90. 出口参数: 无
  91. 说明:LeftSpeed,RightSpeed分别设置左右车轮转速
  92. **********************************************/
  93. //void back_run(unsigned char LeftSpeed,unsigned char RightSpeed)
  94. //{
  95. //        pwm_left = LeftSpeed,pwm_right =  RightSpeed;//设置速度
  96. //        right_motor_back;//右电机后退
  97. //        left_motor_back; //左电机后退
  98. //}
  99. /*********************************************
  100. QX_A11智能小车停车
  101. 入口参数:无
  102. 出口参数: 无
  103. 说明:QX_A11智能小车停车
  104. **********************************************/
  105. void stop(void)
  106. {
  107.         left_motor_stops;
  108.         right_motor_stops;
  109. }

  110. /*====================================
  111. 函数:void Delay_Ms(INT16U ms)
  112. 参数:ms,毫秒延时形参
  113. 描述:12T 51单片机自适应主时钟毫秒级延时函数
  114. ====================================*/
  115. void Delay_Ms(unsigned int ms)
  116. {
  117.      unsigned int i;
  118.          do{
  119.               i = MAIN_Fosc / 96000;
  120.                   while(--i);   //96T per loop
  121.      }while(--ms);
  122. }

  123. /*舵机方波延时朝向小车正前方*/
  124. void Delay1550us()                //@11.0592MHz
  125. {
  126.         unsigned char i, j;

  127.         i = 3;
  128.         j = 196;
  129.         do
  130.         {
  131.                 while (--j);
  132.         } while (--i);
  133. }

  134. void Delay18450us()                //@11.0592MHz
  135. {
  136.         unsigned char i, j;

  137.         _nop_();
  138.         i = 34;
  139.         j = 16;
  140.         do
  141.         {
  142.                 while (--j);
  143.         } while (--i);
  144. }
  145. /*舵机方波延时向小车右方*/
  146. void Delay600us()                //@11.0592MHz
  147. {
  148.         unsigned char i, j;

  149.         _nop_();
  150.         i = 2;
  151.         j = 15;
  152.         do
  153.         {
  154.                 while (--j);
  155.         } while (--i);
  156. }
  157. void Delay19400us()                //@11.0592MHz
  158. {
  159.         unsigned char i, j;

  160.         _nop_();
  161.         i = 35;
  162.         j = 197;
  163.         do
  164.         {
  165.                 while (--j);
  166.         } while (--i);
  167. }
  168. /*舵机方波延时朝向小车左方*/
  169. void Delay17500us()                //@11.0592MHz
  170. {
  171.         unsigned char i, j;

  172.         i = 32;
  173.         j = 93;
  174.         do
  175.         {
  176.                 while (--j);
  177.         } while (--i);
  178. }
  179. void Delay2500us()                //@11.0592MHz
  180. {
  181.         unsigned char i, j;

  182.         i = 5;
  183.         j = 120;
  184.         do
  185.         {
  186.                 while (--j);
  187.         } while (--i);
  188. }
  189. /*********************************************
  190. QX_A11智能小车PWM输出
  191. 入口参数:无
  192. 出口参数: 无
  193. 说明:装载PWM输出,如果设置全局变量pwm_left,pwm_right分别配置左右输出高电平时间
  194. **********************************************/
  195. void LoadPWM(void)
  196. {
  197.         if(pwm_left > PWM_HIGH_MAX)                pwm_left = PWM_HIGH_MAX;        //如果左输出写入大于最大占空比数据,则强制为最大占空比。
  198.         if(pwm_left < PWM_HIGH_MIN)                pwm_left = PWM_HIGH_MIN;        //如果左输出写入小于最小占空比数据,则强制为最小占空比。
  199.         if(pwm_right > PWM_HIGH_MAX)        pwm_right = PWM_HIGH_MAX;        //如果右输出写入大于最大占空比数据,则强制为最大占空比。
  200.         if(pwm_right < PWM_HIGH_MIN)        pwm_right = PWM_HIGH_MIN;        //如果右输出写入小于最小占空比数据,则强制为最小占空比。
  201.         if(pwm_val_left<=pwm_left)                Left_moto_pwm = 1;  //装载左PWM输出高电平时间
  202.         else Left_moto_pwm = 0;                                                     //装载左PWM输出低电平时间
  203.         if(pwm_val_left>=PWM_DUTY)                pwm_val_left = 0;        //如果左对比值大于等于最大占空比数据,则为零

  204.         if(pwm_val_right<=pwm_right)        Right_moto_pwm = 1; //装载右PWM输出高电平时间
  205.         else Right_moto_pwm = 0;                                                         //装载右PWM输出低电平时间
  206.         if(pwm_val_right>=PWM_DUTY)                pwm_val_right = 0;        //如果右对比值大于等于最大占空比数据,则为零
  207. }
  208. //冒泡排序
  209. void QXMBOT_bubble(unsigned int *a,unsigned char n) /*定义两个参数:数组首地址与数组大小*/
  210. {
  211.         unsigned int i,j,temp;        
  212.         for(i = 0;i < n-1; i++)        
  213.         {        
  214.                 for(j = i + 1; j < n; j++) /*注意循环的上下限*/
  215.                 {
  216.                         if(a[i] > a[j])
  217.                         {
  218.                                 temp = a[i];               
  219.                                 a[i] = a[j];               
  220.                                 a[j] = temp;                        
  221.                         }
  222.                 }
  223.         }

  224. }
  225. /*====================================
  226. 函数名        :QXMBOT_RefreshDistance
  227. 参数        :无
  228. 返回值        :经过冒泡排序后的距离
  229. 描述        :经过5次测距,去除最大值和最小值,取中间3次平均值
  230. 距离单位:毫米
  231. ====================================*/
  232. unsigned int QXMBOT_RefreshDistance()
  233. {
  234.         unsigned char num;
  235.         unsigned int Dist;
  236.         for(num=0; num<5; num++)
  237.         {
  238.                 DistBuf[num] = QXMBOT_GetDistance();
  239.                 Delay_Ms(60);//测距周期不低于60毫秒        
  240.         }
  241.         QXMBOT_bubble(DistBuf, 5);//
  242.         Dist = (DistBuf[1]+DistBuf[2]+DistBuf[3])/3; //去掉最大和最小取中间平均值
  243.         return(Dist);
  244. }
  245. /*超声波触发*/
  246. void QXMBOT_TrigUltrasonic()
  247. {
  248.         TrigPin = 0; //超声波模块Trig        控制端
  249.         _nop_();
  250.         _nop_();
  251.         _nop_();
  252.         _nop_();
  253.         _nop_();
  254.         TrigPin = 1; //超声波模块Trig        控制端
  255.         _nop_();_nop_();_nop_();_nop_();_nop_();
  256.         _nop_();_nop_();_nop_();_nop_();_nop_();
  257.         _nop_();_nop_();_nop_();_nop_();_nop_();
  258.         TrigPin = 0; //超声波模块Trig        控制端
  259. }
  260. /*====================================
  261. 函数名        :QXMBOT_GetDistance
  262. 参数        :无
  263. 返回值        :获取距离单位毫米
  264. 描述        :超声波测距
  265. 通过发射信号到收到回响信号的时间测试距离
  266. 单片机晶振11.0592Mhz
  267. 注意测距周期为60ms以上
  268. ====================================*/
  269. unsigned int QXMBOT_GetDistance()
  270. {
  271.         unsigned int Distance = 0;        //超声波距离
  272.         unsigned int  Time=0;                //用于存放定时器时间值
  273.         QXMBOT_TrigUltrasonic();        //超声波触发
  274.         while(!EchoPin);          //判断回响信号是否为低电平
  275.         Timer1On;                        //启动定时器1
  276.         while(EchoPin);                //等待收到回响信号
  277.         Timer1Off;                        //关闭定时器1
  278.         Time=TH1*256+TL1;        //读取时间
  279.         TH1=0;
  280.         TL1=0;                                //清空定时器
  281.     Distance = (float)(Time*1.085)*0.17;//算出来是MM
  282.         return(Distance);//返回距离                                
  283. }
  284. /*====================================
  285. 函数名        :QXMBOT_PTZ_Avoid
  286. 参数        :val设置避障触发距离
  287. 返回值        :无
  288. 描述        :智能小车舵机云台避障
  289. 距离单位:毫米
  290. ====================================*/
  291. void  QXMBOT_PTZ_Avoid(unsigned int val)
  292. {
  293.         unsigned int Dis;//距离暂存变量
  294.         Dis = QXMBOT_GetDistance();//获取超声波测距距离,单位:毫米
  295.         if((Dis > 30) && (Dis < val))
  296.         {
  297.                 LED1=0,LED2=0,beep=0;//LED1,2点亮,鸣笛        
  298.                 stop();        //停车
  299.                 Delay_Ms(100);
  300.                 LED1=1,LED2=1,beep=1;//LED1,2熄灭,静音

  301.                 /*舵机左转测距*/
  302.                 QXMBOT_ServoLeft();
  303.                 LeftDistance = QXMBOT_RefreshDistance();

  304.                 /*舵机右转测距*/
  305.                 QXMBOT_ServoRight();
  306.                 RightDistance = QXMBOT_RefreshDistance();

  307.                 /*舵机正前方测距*/
  308.                 QXMBOT_ServoFront();
  309.                 FrontDistance = QXMBOT_RefreshDistance();
  310.                 if((FrontDistance<100) && (LeftDistance<100) && (RightDistance<100))
  311.                 {
  312.                         do{
  313.                                 left_run(160, 160);//原地左转
  314.                                 Delay_Ms(60);
  315.                                 /*舵机正前方测距*/
  316.                                 QXMBOT_ServoFront();
  317.                                 Dis = QXMBOT_RefreshDistance();               
  318.                         }while(Dis < 200);
  319.                 }else if((FrontDistance>LeftDistance) && (FrontDistance>RightDistance))
  320.                 {
  321.                         stop();        //停车
  322.                         Delay_Ms(100);
  323.                         forward(120, 120);//前进
  324.                 }else if(LeftDistance > RightDistance)
  325.                 {
  326.                         LED1=1,LED2=0;//LED1灭,2点亮
  327.                         stop();        //停车
  328.                         Delay_Ms(100);
  329.                         left_run(160, 160);//原地左转
  330.                         Delay_Ms(60);
  331.                         LED1=1,LED2=1;//LED1灭,2点灭               
  332.                 }else if(RightDistance > LeftDistance)
  333.                 {
  334.                         LED1=0,LED2=1;//LED1亮,2点灭
  335.                         stop();        //停车
  336.                         Delay_Ms(100);
  337.                         right_run(160, 160);//原地右转
  338.                         Delay_Ms(60);
  339.                         LED1=1,LED2=1;//LED1灭,2点灭        
  340.                 }               
  341.         }
  342.         else
  343.         {
  344.                 forward(120, 120);//前进
  345.                 Delay_Ms(60);        
  346.         }                        
  347. }
  348. /*=================================================
  349. *函数名称:QXMBOT_ServoFront
  350. *函数功能:云台向前转动
  351. *调用:
  352. *输入:
  353. =================================================*/
  354. void QXMBOT_ServoFront()
  355. {
  356.         char i;
  357.         EA_off;        //关闭中断否则会影响舵机转向
  358.         for(i=0;i<10;i++)
  359.         {        
  360.                 ServoPin = 1;
  361.                 Delay1550us();
  362.                 ServoPin = 0;
  363.                 Delay18450us();
  364.         }
  365.         EA_on;        //开中断
  366.         Delay_Ms(100);
  367. }
  368. /*=================================================
  369. *函数名称:QXMBOT_ServoLeft
  370. *函数功能:云台向左转动
  371. *调用:
  372. *输入:
  373. =================================================*/
  374. void QXMBOT_ServoLeft()
  375. {
  376.         char i;
  377.         EA_off;        //关闭中断否则会影响舵机转向
  378.         for(i=0;i<10;i++)
  379.         {        
  380.                 ServoPin = 1;
  381.                 Delay2500us();
  382.                 ServoPin = 0;
  383.                 Delay17500us();
  384.         }
  385.         EA_on;        //开中断
  386.         Delay_Ms(100);
  387. }
  388. /*=================================================
  389. *函数名称:QXMBOT_ServoFront
  390. *函数功能:云台向右转动
  391. *调用:
  392. *输入:
  393. =================================================*/
  394. void QXMBOT_ServoRight()
  395. {
  396.         char i;
  397.         EA_off;        //关闭中断否则会影响舵机转向
  398.         for(i=0;i<10;i++)
  399.         {        
  400.                 ServoPin = 1;
  401.                 Delay600us();
  402.                 ServoPin = 0;
  403.                 Delay19400us();
  404.         }
  405.         EA_on;        //开中断
  406.         Delay_Ms(100);
  407. }
  408. /********************* Timer0初始化************************/
  409. void Timer0_Init(void)
  410. {
  411.         TMOD |= 0x02;//定时器0,8位自动重装模块
  412.         TH0 = 164;
  413.         TL0 = 164;//11.0592M晶振,12T溢出时间约等于100微秒
  414.         TR0 = 1;//启动定时器0
  415.         ET0 = 1;//允许定时器0中断        
  416. }
  417. /*定时器1初始化*/
  418. void Timer1_Init(void)               
  419. {
  420.         TMOD |= 0x10;        //定时器1工作模式1,16位定时模式。T1用测ECH0脉冲长度
  421.         TH1 = 0;                  
  422.     TL1 = 0;
  423.         ET1 = 1;             //允许T1中断
  424. }

  425. /********************* Timer0中断函数************************/
  426. void timer0_int (void) interrupt 1
  427. {
  428.          pwm_val_left++;
  429.          pwm_val_right++;
  430.          LoadPWM();//装载PWM输出
  431. }
  432. /* Timer0 interrupt routine */
  433. void tm1_isr() interrupt 3 using 1
  434. {
  435.         Timer1Overflow = 1;        //计数器1溢出标志位
  436.         EchoPin = 0;                //超声波接收端        
  437. }        
复制代码
代码下载: 智能小车I套餐舵机云台避障资料.7z (17.3 MB, 下载次数: 101)

评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

回复

使用道具 举报

ID:143767 发表于 2021-5-5 18:26 | 显示全部楼层
你好楼主,附件里除了程序以外有原理图吗
回复

使用道具 举报

ID:893682 发表于 2021-5-6 20:11 | 显示全部楼层
dj3365191 发表于 2021-5-5 18:26
你好楼主,附件里除了程序以外有原理图吗

有一部分
回复

使用道具 举报

ID:920801 发表于 2021-6-21 11:42 | 显示全部楼层
这个有流程图不?
回复

使用道具 举报

ID:29972 发表于 2021-6-22 22:26 | 显示全部楼层
擦墙现象有吗,有解决方案吗。
回复

使用道具 举报

ID:1024234 发表于 2022-5-6 09:48 | 显示全部楼层
你好,请问可以提供一下 <QX_A11.h>//QX_A11智能小车配置文件 这个文件的代码吗? 新手币不够,下载不了
回复

使用道具 举报

ID:984610 发表于 2022-5-7 23:08 | 显示全部楼层
这个是需要用KeilMDK编辑吗大佬?
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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