找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 1267|回复: 2
收起左侧

两路4轮舵机避障小车的单片机程序,已优化,自行改造

[复制链接]
ID:1091382 发表于 2023-8-28 16:21 | 显示全部楼层 |阅读模式
两路小车改的四轮小车,原先的两轮转向极不稳定,行走也跑偏,不好玩,后来改的四轮,升级7.4v锂电动力,程序里面调速前进(20,20),行走、转向很稳定。原理都一样,无非多两个tt电机,左右轮电机线分别接一起。程序优化了一下能完美避障,距离太近还能自动后退。
程序需要完整下载,里面有个H文件一起加c语言才能用。单独复制没有定义引脚烧录不了的。
制作出来的实物图如下:
4.jpg 6.jpg 1.jpg 2.jpg 3.jpg

单片机源程序如下:
  1. /*********************************************************************************
  2. * 【外部晶振】: 11.0592mhz        
  3. * 【主控芯片】: STC89C52
  4. * 【编译环境】: Keil μVisio4
  5. *        Copyright(C) QXMBOT
  6. *        All rights reserved
  7. ***********************************************************************************
  8. * 【接线说明】:请参考资料内图文说明书                                                                                   
  9. * 【注意事项】:避免小车撞向障碍物或小车轮子堵转
  10. **********************************************************************************/
  11. #include <reg52.h>//51头文件
  12. #include <QX_A11.h>//QX_A11智能小车配置文件
  13. #include <intrins.h>
  14. bit Timer1Overflow;        //计数器1溢出标志位
  15. unsigned char disbuff[4]={0,0,0,0};//用于分别存放距离的值米,厘米,毫米
  16. unsigned int LeftDistance = 0, RightDistance = 0, FrontDistance = 0; //云台测距距离缓存
  17. unsigned int DistBuf[5] = {0};//distance data buffer
  18. unsigned char        pwm_val_left,pwm_val_right;        //中间变量,只是此处用户请勿修改。
  19. unsigned char         pwm_left,pwm_right;                        //定义PWM输出高电平的时间的变量。用户操作PWM的变量。左右()里面的值
  20. #define                PWM_DUTY                200                        //(用于舵机时不可修改)定义PWM的周期,数值为定时器0溢出周期,假如定时器溢出时间为100us,则PWM周期为20ms。
  21. #define                PWM_HIGH_MIN        70                        //限制PWM输出的最小占空比。用户请勿修改。
  22. #define                PWM_HIGH_MAX        PWM_DUTY        //限制PWM输出的最大占空比。用户请勿修改。

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

  46. /*主函数*/     
  47. void main(void)
  48. {
  49.         Timer0_Init();//定时0初始化
  50.         Timer1_Init();//定时0初始化
  51.         EA_on;        //开中断
  52.         QXMBOT_ServoFront();//舵机向前
  53.         Delay_Ms(1000);
  54.         forward(20,50);//前进,forward全部调成(20,20)可调速。7.4v情况下
  55.         while(1)
  56.         {
  57.                 QXMBOT_PTZ_Avoid(300);//舵机云台避障,LCD1602显示距离,设置触发距离300毫米
  58.         }        
  59. }


  60. /*********************************************
  61. QX_A11智能小车前进
  62. 入口参数:LeftSpeed,RightSpeed
  63. 出口参数: 无
  64. 说明:LeftSpeed,RightSpeed分别设置左右车轮转速
  65. **********************************************/
  66. void forward(unsigned char LeftSpeed,unsigned char RightSpeed)
  67. {
  68.         pwm_left = LeftSpeed,pwm_right =  RightSpeed;//设置速度
  69.         left_motor_go; //左电机前进
  70.         right_motor_go; //右电机前进
  71. }
  72. /*小车左转*/
  73. /*********************************************
  74. QX_A11智能小车左转
  75. 入口参数:LeftSpeed,RightSpeed
  76. 出口参数: 无
  77. 说明:LeftSpeed,RightSpeed分别设置左右车轮转速
  78. **********************************************/
  79. void left_run(unsigned char LeftSpeed,unsigned char RightSpeed)
  80. {
  81.         pwm_left = LeftSpeed,pwm_right =  RightSpeed;//设置速度
  82.         left_motor_back; //左电机后退
  83.         right_motor_go; //右电机前进        
  84. }

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

  119. /*====================================
  120. 函数:void Delay_Ms(INT16U ms)
  121. 参数:ms,毫秒延时形参
  122. 描述:12T 51单片机自适应主时钟毫秒级延时函数
  123. ====================================*/
  124. void Delay_Ms(unsigned int ms)
  125. {
  126.      unsigned int i;
  127.          do{
  128.               i = MAIN_Fosc / 96000;
  129.                   while(--i);   //96T per loop
  130.      }while(--ms);
  131. }

  132. /*舵机方波延时朝向小车正前方*/
  133. void Delay1550us()                //@11.0592MHz
  134. {
  135.         unsigned char i, j;

  136.         i = 3;
  137.         j = 196;
  138.         do
  139.         {
  140.                 while (--j);
  141.         } while (--i);
  142. }

  143. void Delay18450us()                //@11.0592MHz
  144. {
  145.         unsigned char i, j;

  146.         _nop_();
  147.         i = 34;
  148.         j = 16;
  149.         do
  150.         {
  151.                 while (--j);
  152.         } while (--i);
  153. }
  154. /*舵机方波延时向小车右方*/
  155. void Delay600us()                //@11.0592MHz
  156. {
  157.         unsigned char i, j;

  158.         _nop_();
  159.         i = 2;
  160.         j = 15;
  161.         do
  162.         {
  163.                 while (--j);
  164.         } while (--i);
  165. }
  166. void Delay19400us()                //@11.0592MHz
  167. {
  168.         unsigned char i, j;

  169.         _nop_();
  170.         i = 35;
  171.         j = 197;
  172.         do
  173.         {
  174.                 while (--j);
  175.         } while (--i);
  176. }
  177. /*舵机方波延时朝向小车左方*/
  178. void Delay17500us()                //@11.0592MHz
  179. {
  180.         unsigned char i, j;

  181.         i = 32;
  182.         j = 93;
  183.         do
  184.         {
  185.                 while (--j);
  186.         } while (--i);
  187. }
  188. void Delay2500us()                //@11.0592MHz
  189. {
  190.         unsigned char i, j;

  191.         i = 5;
  192.         j = 120;
  193.         do
  194.         {
  195.                 while (--j);
  196.         } while (--i);
  197. }
  198. /*********************************************
  199. QX_A11智能小车PWM输出
  200. 入口参数:无
  201. 出口参数: 无
  202. 说明:装载PWM输出,pwm_left,pwm_right分别配置左右输出高电平时间,请设置全局对应行走方向(x,y)数据一致调速
  203. **********************************************/
  204. void LoadPWM(void)
  205. {
  206.         if(pwm_left > PWM_HIGH_MAX)                pwm_left = PWM_HIGH_MAX;        //如果左输出写入大于最大占空比数据,则强制为最大占空比。
  207.         if(pwm_left < PWM_HIGH_MIN)                pwm_left = PWM_HIGH_MIN;        //如果左输出写入小于最小占空比数据,则强制为最小占空比。
  208.         if(pwm_right > PWM_HIGH_MAX)        pwm_right = PWM_HIGH_MAX;        //如果右输出写入大于最大占空比数据,则强制为最大占空比。
  209.         if(pwm_right < PWM_HIGH_MIN)        pwm_right = PWM_HIGH_MIN;        //如果右输出写入小于最小占空比数据,则强制为最小占空比。
  210.         if(pwm_val_left<=pwm_left)                Left_moto_pwm = 1;  //装载左PWM输出高电平时间
  211.         else Left_moto_pwm = 0;                                                     //装载左PWM输出低电平时间
  212.         if(pwm_val_left>=PWM_DUTY)                pwm_val_left = 0;        //如果左对比值大于等于最大占空比数据,则为零

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

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

  310.                 /*舵机左转测距*/
  311.                 QXMBOT_ServoLeft();
  312.                 LeftDistance = QXMBOT_RefreshDistance();

  313.                 /*舵机右转测距*/
  314.                 QXMBOT_ServoRight();
  315.                 RightDistance = QXMBOT_RefreshDistance();

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

  440. /********************* Timer0中断函数************************/
  441. void timer0_int (void) interrupt 1
  442. {
  443.          pwm_val_left++;
  444.          pwm_val_right++;
  445.          LoadPWM();//装载PWM输出
  446. }
  447. /* Timer0 interrupt routine */
  448. void tm1_isr() interrupt 3 using 1
  449. {
  450.         Timer1Overflow = 1;        //计数器1溢出标志位
  451.         EchoPin = 0;                //超声波接收端        
  452. }   
复制代码
   


请下载压缩包解压。
两路4轮舵机避障小车程序.7z (818.55 KB, 下载次数: 23)

评分

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

查看全部评分

回复

使用道具 举报

ID:435636 发表于 2023-8-29 07:46 | 显示全部楼层
谢谢分享,可有材料清单?
回复

使用道具 举报

ID:1091382 发表于 2023-9-4 17:07 | 显示全部楼层
有的材料清单
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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