基于51单片机的超声波避障小车设计(含Proteus仿真)

╰半橙微兮° 2023-10-07 12:09 176阅读 0赞

超声波避障程序随处可见,基于51单片机的超声波避障小车也很成熟,但是完整的Proteus仿真并不容易找到开源资料。

这次主要给大家分享其Proteus仿真部分。

涉及到的模块有:超声波模块(hc-sr04)、L293D电机驱动器和直流减速电机。这/样配合51单片机的控制,小车可以完成自主避障功能。

超声波模块

在这里插入图片描述
此图为Proteus 8 提供的超声波模块(SRF04),它有5个引脚,其中GND接地、VCC接高电平、NC可不接。TR用作激发信号的输入,当超声波模块在TR引脚上检测到了连续的10us以上的高电平时,超声波模块才开始工作。ECHO用作反馈信号输出,当超声波检测到有障碍物时,从该引脚输出相应信号。

电机驱动模块

在这里插入图片描述
上图为Proteus 8提供的电机驱动模块(L293D),4个IN 引脚与单片机连接,控制电机转动及方向,2个使EN 能引脚同样与单片机连接。4个OUT,连接两个直流电机。VSS引脚与VS引脚接高电平即可。

加载程序

双击添加的AT89C51单片机,出现如下对话框。
在这里插入图片描述

点击 Program File 此行文件夹图标,添加HEX文件(keil软件编写程序后编译生成)
在这里插入图片描述
点击运行即可。
在这里插入图片描述

示波器

在这里插入图片描述
示波器可以辅助我们调试程序和仿真,关于在Proteus里调用示波器以及示波器的使用,我不做介绍,很多资料都能查到,本例中我运用示波器观察超声波模块的TR引脚和ECHO引脚的波形。

超声波模块原理图

在这里插入图片描述

电机驱动模块原理

在这里插入图片描述

单片机最小系统

在这里插入图片描述

总原理图

在这里插入图片描述

51程序

  1. #include <at89x51.h>
  2. #include <intrins.h
  3. #define TX P1_3
  4. #define RX P1_2
  5. #define Forward_L_DATA 180 //当前进不能走直线时,调节这两个参数,理想是100,100,最大时256,最小是0.
  6. #define Forward_R_DATA 180
  7. sbit L293D_IN1=P0^0;
  8. sbit L293D_IN2=P0^1;
  9. sbit L293D_IN3=P0^2;
  10. sbit L293D_IN4=P0^3;
  11. sbit L293D_EN1=P0^4;
  12. sbit L293D_EN2=P0^5;
  13. void Delay400Ms(void);//延时400毫秒函数
  14. unsigned char disbuff[4]={
  15. 0,0,0,0};//用于分别存放距离的值0.1mm,mm,cm,m
  16. void Count(void);//距离计算函数
  17. unsigned int time=0;//用于存放定时器的时间值
  18. unsigned long S=0;//用于存放距离的值
  19. bit flag =0;//量程溢出标志位
  20. bit turn_right_flag;
  21. void Delay1ms(unsigned int i)
  22. {
  23. unsigned char j,k;
  24. do{
  25. j = 10;
  26. do{
  27. k = 50;
  28. do{
  29. _nop_();
  30. }while(--k);
  31. }while(--j);
  32. }while(--i);
  33. }
  34. void Delay10us(unsigned char i)
  35. {
  36. unsigned char j;
  37. do{
  38. j = 10;
  39. do{
  40. _nop_();
  41. }while(--j);
  42. }while(--i);
  43. }
  44. void Forward()//前进
  45. {
  46. L293D_IN1=1;
  47. L293D_IN2=0;
  48. L293D_IN3=1;
  49. L293D_IN4=0;
  50. }
  51. void Stop(void)//刹车
  52. {
  53. L293D_IN1=0;
  54. L293D_IN2=0;
  55. L293D_IN3=0;
  56. L293D_IN4=0;
  57. }
  58. void Turn_Retreat()//后
  59. {
  60. L293D_IN1=0;
  61. L293D_IN2=1;
  62. L293D_IN3=0;
  63. L293D_IN4=1;
  64. }
  65. void Turn_left()//左
  66. {
  67. L293D_IN1=0;
  68. L293D_IN2=1;
  69. L293D_IN3=1;
  70. L293D_IN4=0;
  71. }
  72. void Conut(void)//计算距离
  73. {
  74. time=TH1*256+TL1;
  75. TH1=0;
  76. TL1=0;
  77. S=time*2;
  78. S=S*0.17;
  79. if(S<=300)
  80. {
  81. if(turn_right_flag!=1)
  82. {
  83. Stop();
  84. Delay1ms(5);
  85. }
  86. turn_right_flag=1;
  87. P1_7=0;
  88. P2_0=0;
  89. P0_6=0;
  90. Delay1ms(10);
  91. P1_7=1;
  92. P2_0=1;
  93. P0_6=1;
  94. Delay1ms(5);
  95. Turn_left();
  96. Delay1ms(10);
  97. }
  98. else
  99. {
  100. turn_right_flag=0;
  101. Forward();
  102. }
  103. if((S>=5000)||flag==1)//超出测量范围
  104. {
  105. flag=0;
  106. }
  107. else
  108. {
  109. disbuff[0]=S%10;
  110. disbuff[1]=S/10%10;
  111. disbuff[2]=S/100%10;
  112. disbuff[3]=S/1000;
  113. }
  114. }
  115. void zd0() interrupt 3//T0中断用来计数器溢出,超过测距范围
  116. {
  117. flag=1;
  118. RX=0;
  119. }
  120. void Timer_Count(void)
  121. {
  122. TR1=1;//开启计数
  123. while(RX);//当RX为1计数并等待
  124. TR1=0;//关闭计数
  125. Conut();//计算
  126. }
  127. void StartModule()//启动模块
  128. {
  129. TX=1;//启动一次模块
  130. Delay10us(2);
  131. TX=0;
  132. }
  133. void main(void)
  134. {
  135. unsigned char i;
  136. unsigned int a;
  137. Delay1ms(400);
  138. Delay1ms(5);
  139. TMOD=TMOD|0x10;
  140. EA=1;
  141. TH1=0;
  142. TL1=0;
  143. ET1=1;
  144. turn_right_flag=0;
  145. B: for(i=0;i<50;i++)//判断k3是否按下
  146. {
  147. Delay1ms(1);
  148. if(P3_2!=0 )
  149. goto B;
  150. }
  151. while(1)
  152. {
  153. RX=1;
  154. StartModule();
  155. for(a=951;a>0;a--)
  156. {
  157. if(RX==1)
  158. {
  159. Timer_Count();
  160. }
  161. }
  162. }
  163. }

分享决定高度,学习拉开差距

作为学习者给大家分享自己完成的此作品,希望对大家有帮助,当然上文若有不妥之处,欢迎指正。

欢迎大家留言,批评指正!

发表评论

表情:
评论列表 (有 0 条评论,176人围观)

还没有评论,来说两句吧...

相关阅读