循迹小车
硬件介绍和接线
TCRT5000
使用方法和原理见89C52时期的介绍。
循迹小车需要使用两个TCRT5000,左侧的DO接到PB3;右侧的DO接到PB4
CubeMX
1. 在上节的基础上进行修改 + 配置两个传感器的GPIO
2. 惯例配置更新代码
Keil
注意,如果要使用PWM调速就必须全部是PWM调速,因为一旦使用PWM调速之后又使用其他motor.c中的前后左右,就会重置PB1和PB10的值,而如果要使用调速,PB1和PB10就必须一直为HIGH,使用PWM调速的优点是转弯平滑,缺点是电机很容易因为PWM波的有效电平持续时间过低而转不动。
如果不想用PWM调速,不仅不能调用PWM调速的函数,在main中也记得将PWM使能关闭,在Cube中要把PA1和PA2改回GPIO,不使用PWM调速的优点是电机动力充足,缺点是转弯不太平滑。
按照代码封装的方法,封装route_sensor.c和route_sensor.h:
route_sensor.c:
#include "route_sensor.h"
#include "motor.h"
#include "gpio.h"void route_sensor_mode()
{if((HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_3) == GPIO_PIN_RESET) && (HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_4) == GPIO_PIN_RESET)){ //此时左右都没有检测到黑线,说明在轨迹上,直走, 两个灯都亮move_forward();//speed_left(180); //speed就是CCRx的值,理论在0到199之间,实际最好大于130//speed_right(180);}if((HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_3) == GPIO_PIN_SET) && (HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_4) == GPIO_PIN_RESET)){ //此时左模块检测到黑线,说明车身偏右即将离开轨道,此时需要左转修正,只有右灯亮move_leftturn();//speed_left(160); //speed就是CCRx的值,理论在0到199之间,实际最好大于130//speed_right(199);}if((HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_3) == GPIO_PIN_RESET) && (HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_4) == GPIO_PIN_SET)){ //此时右模块检测到黑线,说明车身偏左即将离开轨道,此时需要右转修正,只有左灯亮move_rightturn();//speed_left(199); //speed就是CCRx的值,理论在0到199之间,实际最好大于130//speed_right(160);}if((HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_3) == GPIO_PIN_SET) && (HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_4) == GPIO_PIN_SET)){ //此时左右都检测到黑线,说明不在轨迹上了,停止,两个灯都灭move_stop();//speed_left(0); //speed就是CCRx的值,理论在0到199之间,实际最好大于130//speed_right(0);}}
route_sensor.h:
#ifndef __route_sensor_H__ // "XXXX"就是h文件的名字
#define __route_sensor_H__void route_sensor_mode(void);#endif
main.c:
#include "motor.h"
#include "route_sensor.h"int main(void)
{//HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_1); //打开Timer2的1号Channel, 1号通道对应 左轮//HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_2); //打开Timer2的2号Channel, 2号通道对应 右轮while (1){route_sensor_mode();}
}
实现效果
实现效果和89C52时一样,所以不重复展示了。
循迹小车
硬件介绍和接线
红外避障模块
使用方法和原理见89C52时期的介绍。
跟随小车需要使用两个红外避障模块,左侧的DO接到PB5;右侧的DO接到PB6
CubeMX
1. 在上节的基础上进行修改 + 配置两个传感器的GPIO
2. 惯例配置更新代码
Keil
按照代码封装的方法,封装route_follow.c和route_follow.h:
route_follow.c:
#include "route_follow.h"
#include "motor.h"
#include "gpio.h"void route_follow_mode()
{if((HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_5) == GPIO_PIN_RESET) && (HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_6) == GPIO_PIN_RESET)){ //此时左右都没有检测到黑线,说明在轨迹上,直走, 两个灯都亮move_forward();}if((HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_5) == GPIO_PIN_SET) && (HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_6) == GPIO_PIN_RESET)){ //此时左模块检测到黑线,说明车身偏右即将离开轨道,此时需要左转修正,只有右灯亮move_rightturn();}if((HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_5) == GPIO_PIN_RESET) && (HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_6) == GPIO_PIN_SET)){ //此时右模块检测到黑线,说明车身偏左即将离开轨道,此时需要右转修正,只有左灯亮move_leftturn();}if((HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_5) == GPIO_PIN_SET) && (HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_6) == GPIO_PIN_SET)){ //此时左右都检测到黑线,说明不在轨迹上了,停止,两个灯都灭move_stop();}}
route_follow.h:
#ifndef __route_follow_H__ // "XXXX"就是h文件的名字
#define __route_follow_H__void route_follow_mode(void);#endif
main.c:
#include "motor.h"
#include "route_follow.h"int main(void)
{//HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_1); //打开Timer2的1号Channel, 1号通道对应 左轮//HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_2); //打开Timer2的2号Channel, 2号通道对应 右轮while (1){route_follow_mode();}
}
实现效果
实现效果和89C52时一样,所以不重复展示了。
摇头避障小车
硬件介绍和接线
SG90舵机(实现摇头)
HC-SR04超声波测距模块(实现避障)
由于驱动舵机使用的PWM也是周期为20ms的,所以可以直接用TIM1的CH3,对应PA2
HC-SR04的Trig接PA5;Echo接PA4
CubeMX
1. 在上节的基础上进行修改 + 配置Trig和Echo的GPIO:
2. 打开TIM2的CH3的PWM,由于PSC和ARR已经设置过了所以不用设置,有效电平为HIGH也不用设置:
3. 打开TIM3,用作HC-SR04的计时器,并且只用来作为计数功能,并软件控制何时停止计数,因此只需要设置TIM2的PSC而不需要设置ARR。
神奇的事情又发生了...在前几个项目里,同样是HC-SR04的初始化,使用的是微秒级的延时,但是在这个项目里,又变回了毫秒级的延时???唯一的区别是,上个项目使用的是TIM2,此处是TIM3...
4. 惯例配置更新代码
Keil
按照代码封装的方法,封装SG90.c, SG90.h, HC_SR04.c, HC_SR04.h:
SG90.c:
#include "SG90.h"
#include "tim.h"int angle;void head_right()
{angle = 0;__HAL_TIM_SetCompare(&htim2, TIM_CHANNEL_3, 5); //0度
}void head_middle()
{angle = 1;__HAL_TIM_SetCompare(&htim2, TIM_CHANNEL_3, 15); //90度
}void head_left()
{angle = 2;__HAL_TIM_SetCompare(&htim2, TIM_CHANNEL_3, 25); //180度
}
SG90.h:
#ifndef __SG90_H__ // "XXXX"就是h文件的名字
#define __SG90_H__void head_right(void);
void head_middle(void);
void head_left(void);#endif
HC_SR04.c:
#include "HC_SR04.h"
#include "SG90.h"
#include "gpio.h"
#include "tim.h"
#include "motor.h"double dist_middle;
double dist_left;
double dist_right;void StartHC()
{HAL_GPIO_WritePin(GPIOA, GPIO_PIN_5, GPIO_PIN_RESET); //Trig写0HAL_GPIO_WritePin(GPIOA, GPIO_PIN_5, GPIO_PIN_SET); //Trig写1HAL_Delay(10);HAL_GPIO_WritePin(GPIOA, GPIO_PIN_5, GPIO_PIN_RESET); //Trig写0}double get_dist()
{int cnt;double dist;StartHC();while((HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_4)) == GPIO_PIN_RESET); //等待Echo变高的一瞬间HAL_TIM_Base_Start(&htim3); //TIM3开始计时__HAL_TIM_SetCounter(&htim3,0); //将TIM3的计数器置0while((HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_4)) == GPIO_PIN_SET); //等待Echo变低的一瞬间 HAL_TIM_Base_Stop(&htim3); //TIM3停止计时cnt = __HAL_TIM_GetCounter(&htim3);//求出计了多少次,由于计数一次经过的时间是1usdist = cnt*340/2*0.000001*100; //求出距离return dist;
}void deal_dist()
{dist_middle = get_dist();if(dist_middle > 35){ //如果距离大于35cm就可以前进move_forward();//前进}else if((dist_middle > 10) && (dist_middle < 35)){ //距离小于35cm但大于10cm时move_stop();//先停下head_left();//将测距传感器通过SG90转到左边dist_left = get_dist();HAL_Delay(300);//根据实际情况head_right();//将测距传感器通过SG90转到右边dist_right = get_dist();HAL_Delay(300);//根据实际情况if(dist_right > dist_left){ //哪边距离大往哪边转move_rightturn();HAL_Delay(300);//根据实际情况move_stop();}else{move_leftturn();HAL_Delay(300);//根据实际情况move_stop();}}else{ //当距离已经小于10cm时move_backward();//此时距离已经危险了,需要后退一下HAL_Delay(300);//根据实际情况move_stop();}
}
HC_SR04.h:
#ifndef __HC_SR04_H__ // "XXXX"就是h文件的名字
#define __HC_SR04_H__void deal_dist(void);#endif
main.c:
#include "SG90.h"
#include "HC_SR04.h"extern int angle;int main(void)
{ //HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_1); //打开Timer2的1号Channel, 1号通道对应 左轮//HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_2); //打开Timer2的2号Channel, 2号通道对应 右轮HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_3); //打开Timer2的3号Channel, 控制SG90head_middle();while (1){if(angle != 1){ //如果不在中间的话,回到中间head_middle();HAL_Delay(300);}deal_dist();HAL_Delay(50); //重要}
}
实现效果
实现效果和89C52时一样,所以不重复展示了。