当前位置:   article > 正文

STM32智能小车学习笔记(避障、循迹、跟随)_stm32f103rct6小车结合串口的避障循迹代码

stm32f103rct6小车结合串口的避障循迹代码

我们使用的是STM32CubeMX软件和MDK5

芯片使用的是STM32F103C8T6

完成对STM32CubeMX的初始化后开始我们的第一步点亮一个LED灯

一、点亮LED灯

点亮PC13连接的灯

打开STM32CubeMX软件,pc13设置为输出模式

然后按照这样配置,user label 设置成这个IO口代表名字即可

点击这个生成代码

STM32CubeMX给我们每一个引脚都在main.h里面设置以宏的形式,我们写的代码要放在BEGIN 和END之间。

添加以下代码

  1. HAL_GPIO_TogglePin(LED_GPIO_Port, LED_Pin);
  2. HAL_Delay(100);

编译结束0警告0错误说明我们的格式没出现错误,表示我们成功使用STM32CubeMX配置点亮LED灯。

二、按键控制

原理图

KEY1--PB4         上升沿触发         下拉输入

KEY2--PA12        下降沿触发        上拉输入

 PB4和PA12按照这样进行配置

 使能外部中断,生成代码

 

 重定义中断回调函数,并把点亮LED灯的代码给注释掉。

  1. void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
  2. {
  3. if(GPIO_Pin == KEY1_Pin)//判断一下那个引脚触发中断
  4. {
  5. HAL_Delay(10);//延时消抖
  6. HAL_GPIO_TogglePin(KEY1_GPIO_Port, KEY1_Pin);
  7. }
  8. if(GPIO_Pin == KEY2_Pin)//判断一下那个引脚触发中断
  9. {
  10. HAL_Delay(10);//延时消抖
  11. HAL_GPIO_TogglePin(KEY2_GPIO_Port, KEY2_Pin);
  12. }
  13. }

三、OLED使用

本实验使用的是优信电子--0.96寸OLED显示液晶屏模块 IIC液晶屏 四引脚

把中景园电子0.96OLED显示屏_STM32F103C8_IIC_V1.0文件里面的OLED文件添加到到我们的工程分组里面并修改一些错误。

 把上面这些部分换成下面的样式

  1. #define OLED_SCLK_Clr() HAL_GPIO_WritePin(OLED_SCL_GPIO_Port, OLED_SCL_Pin, GPIO_PIN_RESET)//设置SCL低电平
  2. #define OLED_SCLK_Set() HAL_GPIO_WritePin(OLED_SCL_GPIO_Port, OLED_SCL_Pin, GPIO_PIN_SET)//设置SCL高电平
  3. #define OLED_SDIN_Clr() HAL_GPIO_WritePin(OLED_SDA_GPIO_Port,OLED_SDA_Pin,GPIO_PIN_RESET)//设置SDA低电平
  4. #define OLED_SDIN_Set() HAL_GPIO_WritePin(OLED_SDA_GPIO_Port,OLED_SDA_Pin,GPIO_PIN_SET)//设置SDA高电平

SDA-PB12        SCL-PA15

初始化IO口为输出模式--上拉输出模式(这个OLED是IIC协议,模拟IIC控制OLED的)

        

在main.c中加入以下代码 测试OLED显示屏

  1. OLED_Init(); //初始化OLED
  2. OLED_Clear();
  3. OLED_ShowCHinese(0,0,0);//中
  4. OLED_ShowCHinese(18,0,1);//景
  5. OLED_ShowCHinese(36,0,2);//园
  6. OLED_ShowCHinese(54,0,3);//电
  7. OLED_ShowCHinese(72,0,4);//子
  8. OLED_ShowCHinese(90,0,5);//科
  9. OLED_ShowCHinese(108,0,6);//技

四、串口实验

用cubemx软件配置,选择USART1        Mode配置为asynchronous(异步)其他的不用修改,生成代码。

在usart.c中重定向printf

  1. /**
  2. * @brief 重定向printf (重定向fputc),
  3. 使用时候记得勾选上魔法棒->Target->UseMicro LIB
  4. 可能需要在C文件加typedef struct __FILE FILE;
  5. 包含这个文件#include "stdio.h"
  6. * @param
  7. * @return
  8. */
  9. int fputc(int ch,FILE *stream)
  10. {
  11. HAL_UART_Transmit(&huart1,( uint8_t *)&ch,1,0xFFFF);
  12. return ch;
  13. }

如果有报错添加        typedef struct __FILE FILE;

在main.c中添加        #include "stdio.h"

用printf函数测试一下是否有误

 五、PWM控制电机

PWMA--PA11        PA11、PA8设置成pwm输出

PWMB--PA8

 由参考手册可知TIM1_CH1和TIM1_CH4复用功能重映射到PA8和PA11

 cudemx软件配置生成代码

预分频值设置为1440-1        自动重装载值设置为100-1

 脉冲时长设置为50(也就是占空比为50%)

因为Cude在生成代码时,有很多外设初始化完成后默认是关闭的。需要我们手动开启。

  1. HAL_TIM_PWM_Start(&htim1, TIM_CHANNEL_1);//开启定时器1 通道1 PWM输出
  2. HAL_TIM_PWM_Start(&htim1, TIM_CHANNEL_4);//开启定时器1 通道4 PWM输出

 启动软件仿真

 下图中d表示的是一个周期的时间2.00433ms(0.002S)那么频率为1/0.002 = 500HZ

 使用这个修改占空比

__HAL_TIM_SET_COMPARE(&htim1, TIM_CHANNEL_1, 40);

六、电机驱动和PWM

实验所用电机为A4950

 

 PA11--PWMA、PA8--PWMB 设置成pwm输出,上一步已经设置好了。

PB3--BIN1        输出模式

PB13--AIN1        输出模式

CudeMx配置生成代码

创建motor.c和.h文件

 小车正方向走电平为低电平反方向走电平为高电平(由A4950电机驱动模块使用手册可知正转接低电平)

motor.c

  1. #include "motor.h"
  2. #include "tim.h"
  3. /*******************
  4. * @brief 设置两个电机的转速和方向
  5. * @param motor1:输入1-100,对应控制电机正方向速度在1%-100%、输入-1-(-100)对应控制电机反方向速度在1%-100%
  6. motor2 原理一样
  7. * @return 无
  8. *
  9. *******************/
  10. void Motor_Set(int motor1, int motor2)
  11. {
  12. //根据参数正负 设置选择方向
  13. if(motor1 < 0) BIN1_SET;
  14. else BIN1_RESET;
  15. if(motor2 < 0) AIN1_SET;
  16. else AIN1_RESET;
  17. if(motor1 < 0)
  18. {
  19. if(motor1 < -99) motor1 = -99; //超过PWM幅值
  20. __HAL_TIM_SET_COMPARE(&htim1, TIM_CHANNEL_1, (100+motor1));//修改定时器1 通道1 Pulse改变占空比
  21. }
  22. else
  23. {
  24. if(motor1 > 99) motor1 = 99;
  25. __HAL_TIM_SET_COMPARE(&htim1, TIM_CHANNEL_1, motor1);//修改定时器1 通道1 Pulse改变占空比
  26. }
  27. if(motor2 < 0)
  28. {
  29. if(motor2 < -99) motor2 = -99;//超过PWM幅值
  30. __HAL_TIM_SET_COMPARE(&htim1, TIM_CHANNEL_4, (100+motor2));//修改定时器1 通道4 Pulse改变占空比
  31. }
  32. else
  33. {
  34. if(motor2 > 99) motor2 = 99;
  35. __HAL_TIM_SET_COMPARE(&htim1, TIM_CHANNEL_4, motor2);//修改定时器1 通道4 Pulse改变占空比
  36. }
  37. }

motor.h

  1. #ifndef MOTOR_H_
  2. #define MOTOR_H_
  3. #include "main.h"
  4. #define AIN1_RESET HAL_GPIO_WritePin(AIN1_GPIO_Port, AIN1_Pin, GPIO_PIN_RESET)//设置AIN1 PB13为低电平
  5. #define AIN1_SET HAL_GPIO_WritePin(AIN1_GPIO_Port, AIN1_Pin, GPIO_PIN_SET)//设置AIN1 PB13为高电平
  6. #define BIN1_RESET HAL_GPIO_WritePin(BIN1_GPIO_Port, BIN1_Pin, GPIO_PIN_RESET)//设置BIN1 PB3为低电平
  7. #define BIN1_SET HAL_GPIO_WritePin(BIN1_GPIO_Port, BIN1_Pin, GPIO_PIN_SET)//设置BIN1 PB3为高电平
  8. void Motor_Set(int motor1, int motor2);
  9. #endif

 进行测试

  1. HAL_Delay(500);
  2. Motor_Set(0,0);

七、编码器测速

这里我们选择TI1和TI2上计数(四倍频)

 由原理图可知AO_A,AO_B以及BO_A,BO_B所连引脚分别为PA0、PA1、PB6、PB7

设置CubeMx

1、设置编码器模式 2、自动重装载值设置为65535 3、TI1 TI2都计数

3、TI1 TI2都计数 4、两个滤波器设置为6

5、打开全局中断 同理设置TI4

6、GPIO引脚设置为上拉 生成代码

 定时器中断定时测速

使用定时器1、2ms进入一次中断,使用中断回调函数

1、设置内部时钟源 

2、使能自动重装载

3、开启更新中断

开启定时器以及中断

  1. HAL_TIM_Encoder_Start(&htim2, TIM_CHANNEL_ALL);//开启定时器2
  2. HAL_TIM_Encoder_Start(&htim4, TIM_CHANNEL_ALL);//开启定时器4
  3. HAL_TIM_Base_Start_IT(&htim2); //开启定时器2 中断
  4. HAL_TIM_Base_Start_IT(&htim4); //开启定时器4 中断
  5. HAL_TIM_Base_Start_IT(&htim1); //开启定时器1 中断

定义两个变量保存编码器计数数值以及两个变量表示速度

  1. short Encoder1Count = 0;//编码器计数器数值
  2. short Encoder2Count = 0;
  3. float Motor1Speed = 0.00;
  4. float Motor2Speed = 0.00;
  5. uint16_t TimerCount = 0;

定时器溢出时间计算公式 

 

  1. /*******************
  2. * @brief 定时器1回调函数
  3. * @param ARR == 99 PSC == 1439
  4. * @return 根据定时器溢出时间计算公式可得0.002s溢出一次
  5. *
  6. *******************/
  7. void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
  8. {
  9. if(htim == &htim1)//htim1 500HZ 2ms 中断一次
  10. {
  11. TimerCount++;
  12. if(TimerCount % 5 == 0)//每10ms执行一次
  13. {
  14. Encoder1Count = (short)__HAL_TIM_GET_COUNTER(&htim4);
  15. Encoder2Count = (short)__HAL_TIM_GET_COUNTER(&htim2);
  16. __HAL_TIM_SET_COUNTER(&htim4,0);
  17. __HAL_TIM_SET_COUNTER(&htim2,0);
  18. Motor1Speed = (float)Encoder1Count*100/9.6/11/4;
  19. Motor2Speed = (float)Encoder2Count*100/9.6/11/4;
  20. TimerCount = 0;
  21. }
  22. }
  23. }

在main.c声明

  1. extern float Motor1Speed ;
  2. extern float Motor2Speed ;

在main.c中输出速度即可 

  1. printf("Motor1Speed:%.2f\r\n",Motor1Speed);
  2. printf("Motor2Speed:%.2f\r\n",Motor2Speed);

八、PID速度控制

【PID算法 - 从入门到实战!】https://www.bilibili.com/video/BV1iP411x71X?vd_source=20e2569dfbc86cd3178a9555d0dd7ac2

使用匿名上位机曲线显示速度波形方便观察数据 

niming.c

  1. #include "niming.h"
  2. #include "main.h"
  3. #include "usart.h"
  4. uint8_t data_to_send[100];
  5. //通过F1帧发送4个uint16类型的数据
  6. void ANO_DT_Send_F1(uint16_t _a, uint16_t _b, uint16_t _c, uint16_t _d)
  7. {
  8. uint8_t _cnt = 0; //计数值
  9. uint8_t sumcheck = 0; //和校验
  10. uint8_t addcheck = 0; //附加和校验
  11. uint8_t i = 0;
  12. data_to_send[_cnt++] = 0xAA;//帧头
  13. data_to_send[_cnt++] = 0xFF;//目标地址
  14. data_to_send[_cnt++] = 0xF1;//功能码
  15. data_to_send[_cnt++] = 8; //数据长度
  16. //单片机为小端模式-低地址存放低位数据,匿名上位机要求先发低位数据,所以先发低地址
  17. data_to_send[_cnt++] = BYTE0(_a);
  18. data_to_send[_cnt++] = BYTE1(_a);
  19. data_to_send[_cnt++] = BYTE0(_b);
  20. data_to_send[_cnt++] = BYTE1(_b);
  21. data_to_send[_cnt++] = BYTE0(_c);
  22. data_to_send[_cnt++] = BYTE1(_c);
  23. data_to_send[_cnt++] = BYTE0(_d);
  24. data_to_send[_cnt++] = BYTE1(_d);
  25. for ( i = 0; i < data_to_send[3]+4; i++)
  26. {
  27. sumcheck += data_to_send[i];//和校验
  28. addcheck += sumcheck;//附加校验
  29. }
  30. data_to_send[_cnt++] = sumcheck;
  31. data_to_send[_cnt++] = addcheck;
  32. HAL_UART_Transmit(&huart1,data_to_send,_cnt,0xFFFF);//这里是串口发送函数
  33. }
  34. //,通过F2帧发送4个int16类型的数据
  35. void ANO_DT_Send_F2(int16_t _a, int16_t _b, int16_t _c, int16_t _d) //F2帧 4个 int16 参数
  36. {
  37. uint8_t _cnt = 0;
  38. uint8_t sumcheck = 0; //和校验
  39. uint8_t addcheck = 0; //附加和校验
  40. uint8_t i=0;
  41. data_to_send[_cnt++] = 0xAA;
  42. data_to_send[_cnt++] = 0xFF;
  43. data_to_send[_cnt++] = 0xF2;
  44. data_to_send[_cnt++] = 8; //数据长度
  45. //单片机为小端模式-低地址存放低位数据,匿名上位机要求先发低位数据,所以先发低地址
  46. data_to_send[_cnt++] = BYTE0(_a);
  47. data_to_send[_cnt++] = BYTE1(_a);
  48. data_to_send[_cnt++] = BYTE0(_b);
  49. data_to_send[_cnt++] = BYTE1(_b);
  50. data_to_send[_cnt++] = BYTE0(_c);
  51. data_to_send[_cnt++] = BYTE1(_c);
  52. data_to_send[_cnt++] = BYTE0(_d);
  53. data_to_send[_cnt++] = BYTE1(_d);
  54. for ( i = 0; i < data_to_send[3]+4; i++)
  55. {
  56. sumcheck += data_to_send[i];
  57. addcheck += sumcheck;
  58. }
  59. data_to_send[_cnt++] = sumcheck;
  60. data_to_send[_cnt++] = addcheck;
  61. HAL_UART_Transmit(&huart1,data_to_send,_cnt,0xFFFF);//这里是串口发送函数
  62. }
  63. //通过F3帧发送2个int16类型和1个int32类型的数据
  64. void ANO_DT_Send_F3(int16_t _a, int16_t _b, int32_t _c ) //F3帧 2个 int16 参数 1个 int32 参数
  65. {
  66. uint8_t _cnt = 0;
  67. uint8_t sumcheck = 0; //和校验
  68. uint8_t addcheck = 0; //附加和校验
  69. uint8_t i=0;
  70. data_to_send[_cnt++] = 0xAA;
  71. data_to_send[_cnt++] = 0xFF;
  72. data_to_send[_cnt++] = 0xF3;
  73. data_to_send[_cnt++] = 8; //数据长度
  74. //单片机为小端模式-低地址存放低位数据,匿名上位机要求先发低位数据,所以先发低地址
  75. data_to_send[_cnt++] = BYTE0(_a);
  76. data_to_send[_cnt++] = BYTE1(_a);
  77. data_to_send[_cnt++] = BYTE0(_b);
  78. data_to_send[_cnt++] = BYTE1(_b);
  79. data_to_send[_cnt++] = BYTE0(_c);
  80. data_to_send[_cnt++] = BYTE1(_c);
  81. data_to_send[_cnt++] = BYTE2(_c);
  82. data_to_send[_cnt++] = BYTE3(_c);
  83. for ( i = 0; i < data_to_send[3]+4; i++)
  84. {
  85. sumcheck += data_to_send[i];
  86. addcheck += sumcheck;
  87. }
  88. data_to_send[_cnt++] = sumcheck;
  89. data_to_send[_cnt++] = addcheck;
  90. HAL_UART_Transmit(&huart1,data_to_send,_cnt,0xFFFF);//这里是串口发送函数
  91. }

niming.h

  1. #ifndef NIMING_H
  2. #define NIMING_H
  3. #include "main.h"
  4. //需要发送16位,32位数据,对数据拆分,之后每次发送单个字节
  5. //拆分过程:对变量dwTemp 去地址然后将其转化成char类型指针,最后再取出指针所指向的内容
  6. #define BYTE0(dwTemp) (*(char *)(&dwTemp))
  7. #define BYTE1(dwTemp) (*((char *)(&dwTemp) + 1))
  8. #define BYTE2(dwTemp) (*((char *)(&dwTemp) + 2))
  9. #define BYTE3(dwTemp) (*((char *)(&dwTemp) + 3))
  10. void ANO_DT_Send_F1(uint16_t, uint16_t _b, uint16_t _c, uint16_t _d);
  11. void ANO_DT_Send_F2(int16_t _a, int16_t _b, int16_t _c, int16_t _d);
  12. void ANO_DT_Send_F3(int16_t _a, int16_t _b, int32_t _c );
  13. #endif

添加测试代码

  1. if(Motor1Speed>3.1) Motor1Pwm--;
  2. if(Motor1Speed<2.9) Motor1Pwm++;
  3. if(Motor2Speed>3.1) Motor2Pwm--;
  4. if(Motor2Speed<2.9) Motor2Pwm++;
  5. Motor_Set(Motor1Pwm,Motor2Pwm);
  6. printf("Motor1Speed:%.2f Motor1Pwm:%d\r\n",Motor1Speed,Motor1Pwm);
  7. printf("Motor2Speed:%.2f Motor2Pwm:%d\r\n",Motor2Speed,Motor2Pwm);
  8. HAL_Delay(10);
  9. //电机速度等信息发送到上位机
  10. //注意上位机不支持浮点数,所以要乘100
  11. ANO_DT_Send_F2(Motor1Speed*100, 3.0*100,Motor2Speed*100,3.0*100);

PID代码

pid.c

  1. #include "pid.h"
  2. //定义一个结构体类型变量
  3. tpid pidMotor1Speed;
  4. //给结构体类型变量赋初值
  5. void PID_init(void)
  6. {
  7. pidMotor1Speed.actual_val = 0.0;
  8. pidMotor1Speed.target_val = 0.00;
  9. pidMotor1Speed.err = 0.0;
  10. pidMotor1Speed.err_last = 0.0;
  11. pidMotor1Speed.err_sum = 0.0;
  12. pidMotor1Speed.kp = 0;
  13. pidMotor1Speed.ki = 0;
  14. pidMotor1Speed.kd = 0;
  15. }
  16. //比例p调节控制函数
  17. float P_realize(tpid * pid, float actual_val)
  18. {
  19. pid->actual_val = actual_val; //传递真实值
  20. pid->err = pid->target_val - pid->actual_val; //当前误差=目标值-真实值
  21. //比例控制调节 输出=Kp*当前误差
  22. pid->actual_val = pid->kp*pid->err;
  23. return pid->actual_val;
  24. }
  25. //比例P 积分I 控制函数
  26. float PI_realize(tpid * pid, float actual_val)
  27. {
  28. pid->actual_val = actual_val; //传递真实值
  29. pid->err = pid->target_val - pid->actual_val; //当前误差=目标值-真实值
  30. pid->err_sum += pid->err;//误差累计值 = 当前误差累计和
  31. //使用PI控制 输出=Kp*当前误差+Ki*误差累计值
  32. pid->actual_val = pid->kp*pid->err + pid->ki*pid->err_sum;
  33. return pid->actual_val;
  34. }
  35. // PID控制函数
  36. float PID_realize(tpid * pid, float actual_val)
  37. {
  38. pid->actual_val = actual_val; //传递真实值
  39. pid->err = pid->target_val - pid->actual_val; //当前误差=目标值-真实值
  40. pid->err_sum += pid->err;//误差累计值 = 当前误差累计和
  41. //使用PID控制 输出 = Kp*当前误差 + Ki*误差累计值 + Kd*(当前误差-上次误差)
  42. pid->actual_val = pid->kp*pid->err + pid->ki*pid->err_sum + pid->kd*(pid->err - pid->err_last);
  43. //保存上次误差: 这次误差赋值给上次误差
  44. pid->err_last = pid->err;
  45. return pid->actual_val;
  46. }

pid.h

  1. #ifndef __PID_H
  2. #define __PID_H
  3. //声明一个结构体类型
  4. typedef struct
  5. {
  6. float target_val;//目标值
  7. float actual_val;//实际值
  8. float err; //当前偏差
  9. float err_last; //上次偏差
  10. float err_sum; //误差累计值
  11. float kp,ki,kd; //比例 积分 微分系数
  12. } tpid;
  13. //声明函数
  14. void PID_init(void);
  15. float P_realize(tpid * pid, float actual_val);
  16. float PI_realize(tpid * pid, float actual_val);
  17. float PID_realize(tpid * pid, float actual_val);
  18. #endif

然后在main函数中调用PID_init()函数,别忘把头文件给包含进去。

使用cJSON方便调参

1、调大堆栈都改为0x800

2、开启串口一的全局中断

  1. __HAL_UART_ENABLE_IT(&huart1,UART_IT_RXNE);
  2. //开启串口1接收中断

中断回调函数

  1. uint8_t Usart1_ReadBuf[256]; //串口1 缓冲数组
  2. uint8_t Usart1_ReadCount = 0; //串口1 接收字节计数
  3. if(__HAL_UART_GET_FLAG(&huart1,UART_FLAG_RXNE))//判断huart1 是否读到字节
  4. {
  5. if(Usart1_ReadCount >= 255) Usart1_ReadCount = 0;
  6. HAL_UART_Receive(&huart1,&Usart1_ReadBuf[Usart1_ReadCount++],1,1000);
  7. }

编写函数用于判断串口是否发送完一帧数据

  1. //判断否接收完一帧数据
  2. uint8_t Usart_WaitReasFinish(void)
  3. {
  4. static uint16_t Usart_LastReadCount = 0; //记录上次的计数值
  5. if(Usart1_ReadCount == 0)
  6. {
  7. Usart_LastReadCount = 0;
  8. return 1;//表示没有在接收数据
  9. }
  10. if(Usart1_ReadCount == Usart_LastReadCount)
  11. {
  12. Usart1_ReadCount = 0;
  13. Usart_LastReadCount = 0;
  14. return 0;//已经接收完成了
  15. }
  16. Usart_LastReadCount = Usart1_ReadCount;
  17. return 2;//表示正在接受中
  18. }

把cJSON的.c.h放到工程中去,并在main函数中加入以下代码

  1. #include "cJSON.h"
  2. #include <string.h>
cJSON *cJsonData ,*cJsonVlaue;

  1. if(Usart_WaitReasFinish() == 0)//是否接收完毕
  2. {
  3. cJsonData = cJSON_Parse((const char *)Usart1_ReadBuf);
  4. if(cJSON_GetObjectItem(cJsonData,"p") !=NULL)
  5. {
  6. cJsonVlaue = cJSON_GetObjectItem(cJsonData,"p");
  7. p = cJsonVlaue->valuedouble;
  8. pidMotor1Speed.kp = p;
  9. }
  10. if(cJSON_GetObjectItem(cJsonData,"i") !=NULL)
  11. {
  12. cJsonVlaue = cJSON_GetObjectItem(cJsonData,"i");
  13. i = cJsonVlaue->valuedouble;
  14. pidMotor1Speed.ki = i;
  15. }
  16. if(cJSON_GetObjectItem(cJsonData,"d") !=NULL)
  17. {
  18. cJsonVlaue = cJSON_GetObjectItem(cJsonData,"d");
  19. d = cJsonVlaue->valuedouble;
  20. pidMotor1Speed.kd = d;
  21. }
  22. if(cJSON_GetObjectItem(cJsonData,"a") !=NULL)
  23. {
  24. cJsonVlaue = cJSON_GetObjectItem(cJsonData,"a");
  25. a = cJsonVlaue->valuedouble;
  26. pidMotor1Speed.target_val =a;
  27. }
  28. if(cJsonData != NULL)
  29. {
  30. cJSON_Delete(cJsonData);//释放空间、但是不能删除cJsonVlaue不然会 出现异常错误
  31. }
  32. memset(Usart1_ReadBuf,0,255);//清空接收buf,注意这里不能使用strlen
  33. }
  34. printf("P:%.3f I:%.3f D:%.3f A:%.3f\r\n",p,i,d,a);

九、PID整定方法

把PID控制函数放到中断里面循环调用定时执行

  1. if(TimerCount % 10 == 0)//每20ms执行一次
  2. {
  3. Motor_Set(PID_realize(&pid1_speed, Motor1Speed), PID_realize(&pid2_speed, Motor2Speed));
  4. TimerCount = 0;
  5. }

  1. //定义一个结构体类型变量
  2. tpid pidMotor1Speed;
  3. tpid pidMotor2Speed;
  4. //给结构体类型变量赋初值
  5. void PID_init(void)
  6. {
  7. pidMotor1Speed.actual_val=0.0;
  8. pidMotor1Speed.target_val=0.00;
  9. pidMotor1Speed.err=0.0;
  10. pidMotor1Speed.err_last=0.0;
  11. pidMotor1Speed.err_sum=0.0;
  12. pidMotor1Speed.kp=0;
  13. pidMotor1Speed.ki=0;
  14. pidMotor1Speed.kd=0;
  15. pidMotor2Speed.actual_val=0.0;
  16. pidMotor2Speed.target_val=0.00;
  17. pidMotor2Speed.err=0.0;
  18. pidMotor2Speed.err_last=0.0;
  19. pidMotor2Speed.err_sum=0.0;
  20. pidMotor2Speed.kp=0;
  21. pidMotor2Speed.ki=0;
  22. pidMotor2Speed.kd=0;
  23. }

十、实现小车前后左右运动

  1. //motorPidSetSpeed(1,2);//向右转弯
  2. //motorPidSetSpeed(2,1);//向左转弯
  3. //motorPidSetSpeed(1,1);//前进
  4. //motorPidSetSpeed(-1,-1);//后退
  5. //motorPidSetSpeed(0,0);//停止
  6. //motorPidSetSpeed(-1,1);//右原地旋转
  7. //motorPidSetSpeed(1,-1);//左原地旋转
  8. void motorPidSetSpeed(float Motor1SetSpeed,float Motor2SetSpeed)
  9. {
  10. //改变电机PID参数的目标速度
  11. pidMotor1Speed.target_val = Motor1SetSpeed;
  12. pidMotor2Speed.target_val = Motor2SetSpeed;
  13. //根据PID计算 输出作用于电机
  14. Motor_Set(PID_realize(&pidMotor1Speed,Motor1Speed),PID_realize(&pidMotor2Speed,Motor2Speed));
  15. }
  16. //向前加速函数
  17. void motorSpeedUp(void)
  18. {
  19. static float MotorSetSpeedUp = 0.5;//静态变量 函数结束变量不会销毁
  20. if(MotorSetSpeedUp <= MAX_SPEED_UP) MotorSetSpeedUp += 0.5; //如果没有超过最大值就增加0.5
  21. motorPidSetSpeed(MotorSetSpeedUp, MotorSetSpeedUp);//设置到电机
  22. }
  23. //向前减速函数
  24. void motorSpeedCut(void)
  25. {
  26. static float MotorSetSpeedCut = 3;//静态变量 函数结束变量不会销毁
  27. if(MotorSetSpeedCut >= 0.5) MotorSetSpeedCut -= 0.5;
  28. motorPidSetSpeed(MotorSetSpeedCut, MotorSetSpeedCut);
  29. }

在main函数写以下代码其中一个就能实现对小车的控制

  1. //motorPidSetSpeed(1,2);//向右转弯
  2. //motorPidSetSpeed(2,1);//向左转弯
  3. //motorPidSetSpeed(1,1);//前进
  4. //motorPidSetSpeed(-1,-1);//后退
  5. //motorPidSetSpeed(0,0);//停止
  6. //motorPidSetSpeed(-1,1);//右原地旋转
  7. //motorPidSetSpeed(1,-1);//左原地旋转

十一、OLED显示速度与历程

  1. /*里程数(cm) += 时间周期(s)*车轮转速(转/s)*车轮周长(cm)*/
  2. Mileage += 0.02*Motor1Speed*22;

  1. extern float Mileage;//里程数
  2. uint8_t OledString[20];
  1. /*******************
  2. * sprintf 函数说明 函数sprintf()用来作格式化的输出
  3. * 函数sprintf()的用法和printf()函数一样,只是sprintf()函数给出第一个参数string(一般为字符数组)
  4. * 一定要在调用sprintf之前分配足够大的空间给buf。
  5. *
  6. *******************/
  7. sprintf((char*)OledString, "V1:%.2fV2:%.2f", Motor1Speed, Motor2Speed);//显示两个电机的速度
  8. OLED_ShowString(0, 0, OledString, 12);//这个是oled驱动里面的,是显示位置的一个函数
  9. sprintf((char*)OledString, "Mileage:%.2f", Mileage);//显示里程数
  10. OLED_ShowString(0, 1, OledString, 12);

十二、OLED显示ADC采集电压

原理图

ADC连接PA4引脚

adc.c

别忘在adc.h中声明电池电压函数

  1. /*******************
  2. * @brief 电池电压测量函数
  3. * @param
  4. * @return 小车电池电压
  5. *
  6. *******************/
  7. float adcGetBatteryVoltage(void)
  8. {
  9. HAL_ADC_Start(&hadc2);//启动ADC转化
  10. if(HAL_OK == HAL_ADC_PollForConversion(&hadc2,50))//等待转化完成、超时时间50ms
  11. return (float)HAL_ADC_GetValue(&hadc2)/4096*3.3*5;//计算电池电压
  12. return -1;
  13. }

在main函数中加

  1. sprintf((char*)OledString, "U:%.2fV", adcGetBatteryVoltage());
  2. OLED_ShowString(0,2,OledString,12);

十三、PID循迹

DO 高电平->有黑线 小灯灭

DO低电平->没有黑线 小灯亮

原理图

OUT_1-PA5、OUT_2-PA7、OUT_3-PB0、OUT_4-PB1

  1. #define READ_HW_OUT_1 HAL_GPIO_ReadPin(HW_OUT_1_GPIO_Port, HW_OUT_1_Pin)
  2. //读取红外对管连接的GPIO电平
  3. #define READ_HW_OUT_2 HAL_GPIO_ReadPin(HW_OUT_2_GPIO_Port, HW_OUT_2_Pin)
  4. #define READ_HW_OUT_3 HAL_GPIO_ReadPin(HW_OUT_3_GPIO_Port, HW_OUT_3_Pin)
  5. #define READ_HW_OUT_4 HAL_GPIO_ReadPin(HW_OUT_4_GPIO_Port, HW_OUT_4_Pin)

 在pid.c中加一下代码

  1. tPid pidHW_Tracking;//红外循迹的PID
  2. pidHW_Tracking.actual_val = 0.0;
  3. pidHW_Tracking.target_val = 0.00;//红外循迹PID 的目标值为0
  4. pidHW_Tracking.err = 0.0;
  5. pidHW_Tracking.err_last = 0.0;
  6. pidHW_Tracking.err_sum = 0.0;
  7. pidHW_Tracking.Kp = -1.50;
  8. pidHW_Tracking.Ki = 0;
  9. pidHW_Tracking.Kd = 0.80;

在main函数中加一下代码

  1. extern tPid pidHW_Tracking;//红外循迹的PID
  2. uint8_t g_ucaHW_Read[4] = {0};//保存红外对管电平的数组
  3. int8_t g_cThisState = 0;//这次状态
  4. int8_t g_cLastState = 0; //上次状态
  5. float g_fHW_PID_Out;//红外对管PID计算输出速度
  6. float g_fHW_PID_Out1;//电机1的最后循迹PID控制速度
  7. float g_fHW_PID_Out2;//电机2的最后循迹PID控制速度
  8. g_ucaHW_Read[0] = READ_HW_OUT_1;//读取红外对管状态、这样相比于写在if里面更高效
  9. g_ucaHW_Read[1] = READ_HW_OUT_2;
  10. g_ucaHW_Read[2] = READ_HW_OUT_3;
  11. g_ucaHW_Read[3] = READ_HW_OUT_4;
  12. if(g_ucaHW_Read[0] == 0 && g_ucaHW_Read[1] == 0 && g_ucaHW_Read[2] == 0 && g_ucaHW_Read[3] == 0)
  13. {
  14. g_cThisState = 0;//前进
  15. }
  16. else if(g_ucaHW_Read[0] == 0 && g_ucaHW_Read[1] == 1 && g_ucaHW_Read[2] == 0 && g_ucaHW_Read[3] == 0)
  17. {
  18. g_cThisState = -1;//右转
  19. }
  20. else if(g_ucaHW_Read[0] == 1 && g_ucaHW_Read[1] == 0 && g_ucaHW_Read[2] == 0 && g_ucaHW_Read[3] == 0)
  21. {
  22. g_cThisState = -2;//快速右转
  23. }
  24. else if(g_ucaHW_Read[0] == 1 && g_ucaHW_Read[1] == 1 && g_ucaHW_Read[2] == 0 && g_ucaHW_Read[3] == 0)
  25. {
  26. g_cThisState = -3;//快速右转
  27. }
  28. else if(g_ucaHW_Read[0] == 0 && g_ucaHW_Read[1] == 0 && g_ucaHW_Read[2] == 1 && g_ucaHW_Read[3] == 0)
  29. {
  30. g_cThisState = 1;//左转
  31. }
  32. else if(g_ucaHW_Read[0] == 0 && g_ucaHW_Read[1] == 0 && g_ucaHW_Read[2] == 0 && g_ucaHW_Read[3] == 1)
  33. {
  34. g_cThisState = 2;//快速左转
  35. }
  36. else if(g_ucaHW_Read[0] == 0 && g_ucaHW_Read[1] == 0 && g_ucaHW_Read[2] == 1 && g_ucaHW_Read[3] == 1)
  37. {
  38. g_cThisState = 3;//快速左转
  39. }
  40. g_fHW_PID_OUT = PID_realize(&pidHW_Tracking, g_cThisState);//PID计算输出目标速度 这个速度,会和基础速度加减
  41. g_fHW_PID_OUT1 = 3 + g_fHW_PID_OUT;//电机1速度=基础速度+循迹PID输出速度
  42. g_fHW_PID_OUT2 = 3 - g_fHW_PID_OUT;//电机1速度=基础速度-循迹PID输出速度
  43. if(g_fHW_PID_OUT1 > 5) g_fHW_PID_OUT1 = 5;//进行限幅 限幅速度在0-5之间
  44. if(g_fHW_PID_OUT1 < 0) g_fHW_PID_OUT1 = 0;
  45. if(g_fHW_PID_OUT2 > 5) g_fHW_PID_OUT2 = 5;
  46. if(g_fHW_PID_OUT2 < 0) g_fHW_PID_OUT2 = 0;
  47. if(g_cThisState != g_cLastState)//如何这次状态不等于上次状态、就进行改变目标速度和控制电机、在定时器中依旧定时控制电机
  48. {
  49. motorPidSetSpeed(g_fHW_PID_OUT1, g_fHW_PID_OUT2);//通过计算的速度控制电机
  50. }
  51. g_cLastState = g_cThisState;//保存上次红外对管状态

十四、手机遥控

原理图

CubeMx配置

1、点击USART3模式选择异步通信

2、打开串口三全局中断

打开串口接收数据

  HAL_UART_Receive_IT(&huart3, &g_ucUsart3ReceiveData, 1);  //串口三接收数据

重定义串口回调函数

uint8_t g_ucUsart3ReceiveData;  //保存串口三接收的数据
  1. //串口接收回调函数
  2. void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
  3. {
  4. if( huart == &huart3)//判断中断源
  5. {
  6. if(g_ucUsart3ReceiveData == 'A') motorPidSetSpeed(1,1);//前运动
  7. if(g_ucUsart3ReceiveData == 'B') motorPidSetSpeed(-1,-1);//后运动
  8. if(g_ucUsart3ReceiveData == 'C') motorPidSetSpeed(0,0);//停止
  9. if(g_ucUsart3ReceiveData == 'D') motorPidSetSpeed(1,2);//右边运动
  10. if(g_ucUsart3ReceiveData == 'E') motorPidSetSpeed(2,1);//左边运动
  11. if(g_ucUsart3ReceiveData == 'F') motorSpeedUp();//加速
  12. if(g_ucUsart3ReceiveData == 'G') motorSpeedCut();//减速
  13. HAL_UART_Receive_IT( &huart3, &g_ucUsart3ReceiveData, 1);//继续进行中断接收
  14. }
  15. }

十五、超声波避障

GPIO工作模式

原理图

具体问题可参考HC_SR04工作原理

Trig(PB5)我们配置为GPIO输出

Echo(PA6)我们配置GPIO输入功能

CubeMx配置

HC_SR04.c

  1. #include "HC_SR04.h"
  2. //因为我们不适用定时器所以我们需要自己写一个us级延时函数
  3. /*******************
  4. * @brief us级延时
  5. * @param usdelay:要延时的us时间
  6. * @return
  7. *
  8. *******************/
  9. void HC_SR04_Delayus(uint32_t usdelay)
  10. {
  11. __IO uint32_t Delay = usdelay * (SystemCoreClock / 8U / 1000U / 1000);//SystemCoreClock:系统频率
  12. do
  13. {
  14. __NOP();
  15. }
  16. while(Delay --);
  17. }
  18. /*******************
  19. * @brief HC_SR04读取超声波距离
  20. * @param 无
  21. * @return 障碍物距离单位:cm (静止表面平整精度更高)
  22. *注意:两个HC_SR04_Read()函数调用的时间间隔要2ms及以上,测量范围更大 精度更高
  23. *******************/
  24. float HC_SR04_Read(void)
  25. {
  26. uint32_t i = 0;
  27. float Distance;
  28. HAL_GPIO_WritePin(HC_SR04_Ting_GPIO_Port, HC_SR04_Ting_Pin, GPIO_PIN_SET);//输出15us高电平
  29. HC_SR04_Delayus(15);
  30. HAL_GPIO_WritePin(HC_SR04_Ting_GPIO_Port, HC_SR04_Ting_Pin, GPIO_PIN_RESET);//高电平输出结束,设置为低电平
  31. while(HAL_GPIO_ReadPin(HC_SR04_Echo_GPIO_Port, HC_SR04_Echo_Pin) == GPIO_PIN_RESET)//等待回响高电平
  32. {
  33. i++;
  34. HC_SR04_Delayus(1);
  35. if(i>10000) return -1;//超时退出循环、防止程序卡死这里
  36. }
  37. i = 0;
  38. while(HAL_GPIO_ReadPin(HC_SR04_Echo_GPIO_Port, HC_SR04_Echo_Pin) == GPIO_PIN_SET)//下面循环是2us
  39. {
  40. i = i+1;
  41. HC_SR04_Delayus(1);//1us 延时,但是整个循环大概是2us左右(因为延时1us 42-44行代码跑也需要一定的时间)
  42. if(i>10000) return -2;//超时退出循环
  43. }
  44. Distance = i*2*0.033/2;//这里乘2的原因是上面的2us
  45. return Distance;
  46. }

HC_SR04.h

  1. #ifndef __HC_SR04_H
  2. #define __HC_SR04_H
  3. #include "main.h"
  4. void HC_SR04_Delayus(uint32_t usdelay);
  5. float HC_SR04_Read(void);
  6. #endif

main函数中加

  1. //避障逻辑
  2. if(HC_SR04_Read() > 25)//前方无障碍
  3. {
  4. motorPidSetSpeed(1,1);//前运动
  5. HAL_Delay(100);
  6. }
  7. else//前方有障碍物
  8. {
  9. motorPidSetSpeed(-1,1);//向右原地转
  10. HAL_Delay(500);
  11. if(HC_SR04_Read() > 25)//右边无障碍
  12. {
  13. motorPidSetSpeed(1,1);//前运动
  14. HAL_Delay(100);
  15. }
  16. else//右边有障碍
  17. {
  18. motorPidSetSpeed(1,-1);//向左原地转
  19. HAL_Delay(1000);
  20. if(HC_SR04_Read() > 25)//左边无障碍
  21. {
  22. motorPidSetSpeed(1,1);//前运动
  23. HAL_Delay(100);
  24. }
  25. else
  26. {
  27. motorPidSetSpeed(-1,-1);//后退
  28. HAL_Delay(1000);
  29. motorPidSetSpeed(-1,1);//右运动
  30. HAL_Delay(50);
  31. }
  32. }
  33. }

十六、超声波(PID)跟随

 pid.c中加以下代码

  1. tPid pidFollow;//定距离跟随PID
  2. pidFollow.actual_val=0.0;
  3. pidFollow.target_val=22.50;
  4. pidFollow.err=0.0;
  5. pidFollow.err_last=0.0;
  6. pidFollow.err_sum=0.0;
  7. pidFollow.kp=-0.5; //定距离跟随的Kp大小通过估算PID输入输出数据,确定大概大小,然后在调试
  8. pidFollow.ki=-0.001;
  9. pidFollow.kd=0;

main.c中加以下代码

  1. extern tpid pidFollow;
  2. float g_fHC_SR04_Read;//超声波传感器读取障碍物数据
  3. float g_fFollow_PID_OUT;//定距离跟随PID计算输出速度
  4. g_fHC_SR04_Read = HC_SR04_Read();//读取前方障碍物距离
  5. if(g_fHC_SR04_Read < 60)//如果前60cm 有东西就启动跟随
  6. {
  7. g_fFollow_PID_OUT = PID_realize(&pidFollow,g_fHC_SR04_Read);//PID计算输出目标速度 这个速度,会和基础速度加减
  8. if(g_fFollow_PID_OUT > 6) g_fFollow_PID_OUT = 6;//对输出速度限幅
  9. if(g_fFollow_PID_OUT < -6) g_fFollow_PID_OUT = -6;
  10. motorPidSetSpeed(g_fFollow_PID_OUT, g_fFollow_PID_OUT);//速度作用与电机上
  11. }
  12. else motorPidSetSpeed(0,0);//如果前面60cm 没有东西就停止
  13. HAL_Delay(10);//读取超声波传感器不能过快

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/天景科技苑/article/detail/767681
推荐阅读
相关标签
  

闽ICP备14008679号