【STM32+HAL】巡逻打靶小车

news2024/9/29 3:26:59

一、前言

作为电赛最爱出的小车和视觉题,将两者结合起来出题也是一个方向,故写下此文供学者参考,也作为备赛电赛的记录。

如有小伙伴想交流学习心得,欢迎加入群聊751950234,群内不定期更新代码,以及提供本人博客所有源码

二、题目分析

实现目标:当小车识别到黑块时,小车停止,开启激光打靶模式,并实现来回巡线打靶。

 思路:为避免小车转头带来的不确定性,故本文采用前后各装一个光电传感器实现前进后退;OpenMV摄像头和激光装载在云台上用作打靶。

三、所用工具

1、芯片: STM32F407ZGT6

2、IDE: MDK-Keil软件

3、库文件:STM32F4xxHAL库

4、视觉模块:OpenMV4

5、云台:幻尔数字舵机云台

四、CubeMX配置

1、定时器配置

TIM1:马达控制定时器,控制周期1ms

TIM6:云台控制定时器,控制周期5ms

TIM2:电机控制PWM生成定时器

TIM5:云台控制PWM生成定时器

TIM3、TIM4:编码器定时器

有关定时器的相关配置,详见 【STM32+HAL】定时器功能小记

2、串口配置

USART1:单片机与电脑通讯

USART2:单片机与OpenMV通讯

至此,CubeMX配置完毕。

五、OpenMV识别

代码不难,大家自行理解。

有关更多OpenMV的内容,详见【OPENMV】学习记录 (持续更新)

import sensor, image, time, ustruct, math
from pyb import UART

uart = UART(3, 115200, timeout_char=200)
uart.init(115200, bits=8, parity=None, stop=1)  # 使用给定参数初始化

# 初始化摄像头
sensor.reset()                        # 初始化感应器
sensor.set_pixformat(sensor.RGB565)   # 设置像素格式为RGB565
sensor.set_framesize(sensor.QVGA)     # 设置帧大小为320x240
sensor.skip_frames(time = 2000)       # 跳过前2秒帧,用于摄像头设置稳定
sensor.set_auto_gain(False)           # 必须关闭才能进行颜色跟踪
sensor.set_auto_whitebal(False)       # 必须关闭才能进行颜色跟踪
clock = time.clock()                  # 初始化时钟对象

def find_circle(blobs):                # 寻找最圆色块
    max_circle = 1
    max_blob = None
    for blob in blobs:
        if blob.elongation() < max_circle:
            max_blob = blob
            max_circle = blob.elongation()
    return max_blob
def send_data(x, y):
    global uart
    uart.write(str(x))
    uart.write(bytearray([0x20]))
    uart.write(str(y))
    uart.write(bytearray([0x20]))

# 设置颜色阈值
threshold = (7, 54, 12, 63, 16, 69)

while(True):
    clock.tick()  # 开始新帧计时
    img = sensor.snapshot().gaussian(1,unsharp=True)  # 捕获图像
    blobs = img.find_blobs([threshold])
    if blobs:
        max_blob = find_circle(blobs)
        if max_blob:
            # 画出最大圆的中心和方向
            img.draw_keypoints([(max_blob.cx(), max_blob.cy(), int(math.degrees(max_blob.rotation())))], size=20, color=(255, 0, 0))
            img.draw_cross(max_blob.cx(), max_blob.cy(), color=(255, 0, 0))
            send_data(max_blob.cx(), max_blob.cy())
            # 打印出最大色块的中心位置和面积
            print(max_blob.cx(), max_blob.cy())

六、Keil填写代码

1、小车循迹函数

其中,P1~5为车前光电传感器,Q1~5为车尾光电传感器。

/*=================== 循迹函数 ===================*/
int Track(void)
{
	int output=0;
	/* 前进差速 */
	if(flag == 1)
	{
		Speed_Middle  = 10;						//中值速度 10
		if(P2) output += 5;
		if(P4) output -= 5;
		if(P1) output += 8;
		if(P5) output -= 8;
	}
	/* 后退差速 */
	else if(flag == 2)
	{
		Speed_Middle  = -8;						//中值速度 -10
		if(Q2) output -= 10;
		else if(Q4) output += 10;
		if(Q1) output -= 15;
		else if(Q5) output += 15;
	}
	/* 直线匀速 */
	if((Q3 && Q2 == GPIO_PIN_RESET && Q4 == GPIO_PIN_RESET)
	 ||(P3 && P2 == GPIO_PIN_RESET && P4 == GPIO_PIN_RESET))
		output = 0;

	return output;
}
2、十字路口判定函数
/*=================== 十字路口判断函数 ===================*/
uint8_t Crossing(void)
{
	/* 前进档 */
	if(flag == 1)
	{
		if((P3 && P2 && P4))		//十字路口识别
		{
			Cross++;
			if(Cross >= 40)			//终点
			{
				Cross = 0;
				flag = 2;
				flag_temp = flag;
				return 1;
			}
			else if(Cross == 20)	//中点
			{
				flag_temp = flag;	//保存运动状态
				stop_cnt = 0;		//打靶时间
				flag = 0;			//暂停小车
				flag_stop = 1;		//开启云台
				return 1;
			}
		}
	}
	/* 倒挡 */
	else if(flag == 2)
	{
		if((Q3 && Q2 && Q4))
		{
			Cross++;
			if(Cross >= 45)
			{
				Cross = 0;
				flag = 1;
				flag_temp = flag;
				return 1;
			}
			else if(Cross == 20)	//中点
			{
				flag_temp = flag;	//保存运动状态
				stop_cnt = 0;		//打靶时间
				flag = 0;			//暂停小车
				flag_stop = 1;		//开启云台
				return 1;
			}
		}
	}
	return 0;
}

3、PID计算函数
/*=================== 增量式PID控制设计 ===================*/
//左A轮PID
float PID_A(float Encoder,float Target)
{
	static float Bias, Last_bias, Last2_bias, Pwm;
	Bias = Target - Encoder;
	Pwm += Kp1 * (Bias - Last_bias) + Ki1 * Bias + Kd1 * (Bias - 2 * Last_bias + Last2_bias);
	Last_bias = Bias;
	Last2_bias = Last_bias;
	return Pwm;
}

//右B轮PID
float PID_B(float Encoder,float Target)
{
	static float Bias, Last_bias, Last2_bias, Pwm;
	Bias = Target-Encoder;
	Pwm += Kp1 * (Bias - Last_bias) + Ki1 * Bias + Kd1 * (Bias - 2 * Last_bias + Last2_bias);
	Last_bias = Bias;
	Last2_bias = Last_bias;
	return Pwm;
}

//打靶PID_X
float PID_Target_X(float now,float target)
{
	static float Bias, Last_bias, Last2_bias, Pwm;
	Bias = target-now;
	Pwm += Kp2 * (Bias - Last_bias) + Ki2 * Bias + Kd2 * (Bias - 2 * Last_bias + Last2_bias);
	Last_bias = Bias;
	Last2_bias = Last_bias;
	return Pwm;
}

//打靶PID_Y
float PID_Target_Y(float now,float target)
{
	static float Bias, Last_bias, Last2_bias, Pwm;
	Bias = target-now;
	Pwm += Kp3 * (Bias - Last_bias) + Ki3 * Bias + Kd3 * (Bias - 2 * Last_bias + Last2_bias);
	Last_bias = Bias;
	Last2_bias = Last_bias;
	return Pwm;
}

4、电机云台控制总函数
/**************************************************************************
Function: Control function
Input   : none
Output  : none
函数功能:控制小车巡线
入口参数:无
返回  值:无
**************************************************************************/
void Control(void)
{
	if(flag == 0)									//暂停模式
		Motor_Left = 0, Motor_Right = 0, Cross = 0;
	else if(Crossing())								//识别到十字路口
		Motor_Left = 0, Motor_Right = 0;
	else											//普通巡线
	{
		CurrentA = (float)Read_Encoder(3);
		CurrentB = (float)Read_Encoder(4);
		TargetA = Speed_Middle + Track();
		TargetB = Speed_Middle - Track();

		Motor_Left  = (int)PWM_Limit(PID_A(CurrentA,TargetA), Limit, -Limit);
		Motor_Right = (int)PWM_Limit(PID_B(CurrentB,TargetB), Limit, -Limit);
	}
	Set_Pwm(Motor_Left, Motor_Right);
}



/**************************************************************************
Function: Target_Control
Input   : none
Output  : none
函数功能:控制激光云台
入口参数:无
返回  值:无
**************************************************************************/
void Target_Control(void)
{
	static int PWMX = 1500, PWMY = 1500;
	PWMX += PID_Target_X(110,Tx);
	PWMY += PID_Target_Y(100,Ty);
	PWMX = (PWMX < 1000) ? 1500 :((PWMX > 2200) ? 1500 : PWMX);
	PWMY = (PWMY < 1000) ? 1500: ((PWMY > 2200) ? 1500 : PWMY);
	PTZA = PWMX;
	PTZB = PWMY;
}

5、main.c

为缩减篇幅,下附代码删减了部分冗余的代码,相信以大家的聪明才智一定看得懂!

int main(void)
{
  /* USER CODE BEGIN 2 */
	/* OPENMV初始化 */
	__HAL_UART_ENABLE_IT(&huart2, UART_IT_IDLE);
	HAL_UART_Receive_DMA(&huart2,rx_buffer,RXBUFFERSIZE);
	HAL_Delay(100);

	/* 定时器初始化 */
	TIM_Init();
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */
    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}
/* USER CODE BEGIN 4 */
/*=================== 定时器中断 ===================*/
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{
	if(flag == 0){
		Set_Pwm(0,0);					//马达停转
		if(htim -> Instance == TIM6)
		{
			if(flag_stop) 				//到中点后暂停,开始识别
			{
				stop_cnt++;
				if(stop_cnt >= 150)		//暂停一段时间后
				{
					flag = flag_temp;	//恢复停止前状态
					stop_cnt = 0;
				}
			}
			Target_Control();
		}
	}
	else
	{
		if (htim -> Instance == TIM1){
			Control();
			PTZA = 1500;				//云台复位
			PTZB = 1500;
		}
	}
}


/*=================== 定时器初始化 ===================*/
void TIM_Init(void)
{
	/* 使能编码器输出 */
	HAL_TIM_Encoder_Start(&htim3,TIM_CHANNEL_ALL);
	HAL_TIM_Encoder_Start(&htim4,TIM_CHANNEL_ALL);

	/* 失能所有输出 */
	AIN10;AIN20;BIN10;BIN20;
	TIM2->CCR1 = 0;
	TIM2->CCR2 = 0;
	TIM5->CCR2 = 1500;
	TIM5->CCR4 = 1500;

	/* 开启PWM波 */
	HAL_TIM_PWM_Start(&htim2,TIM_CHANNEL_1);
	HAL_TIM_PWM_Start(&htim2,TIM_CHANNEL_2);
	HAL_TIM_PWM_Start(&htim5,TIM_CHANNEL_2);
	HAL_TIM_PWM_Start(&htim5,TIM_CHANNEL_4);

	HAL_Delay(1000);
	/* 开启控制中断 */
	HAL_TIM_Base_Start_IT(&htim1);
	HAL_TIM_Base_Start_IT(&htim6);
}


/*=================== 按键切换模式 ===================*/
void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)  
{
	if(HAL_GPIO_ReadPin(WKUP_GPIO_Port,WKUP_Pin) == GPIO_PIN_SET){
		HAL_Delay(20); //延时消抖
		if(HAL_GPIO_ReadPin(WKUP_GPIO_Port,WKUP_Pin) == GPIO_PIN_SET){
			HAL_GPIO_TogglePin(LED_GPIO_Port,LED_Pin);
			flag = (flag + 1) % 3;						//按键切换模式
		}
	}
}
/* USER CODE END 4 */

七、源码提供

夸克网盘:我用夸克网盘分享了「WheeleCar」,点击链接即可保存。

百度网盘:通过百度网盘分享的文件:WheeleCar 提取码:6666

Gitee:WheeleCar

CSDN:WheeleCar

八、结语

本人能力有限,代码未必是最优解,若有可改进之处望在评论区留言。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2052307.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

elasticsearch的高亮查询三种模式查询及可能存在的问题

目录 高亮查询使用介绍 高亮参数 三种分析器 可能存在的查询问题 fvh查询时出现StringIndexOutOfBoundsException越界 检索高亮不正确 参考文档 高亮查询使用介绍 Elasticsearch 的高亮&#xff08;highlight&#xff09;可以从搜索结果中的一个或多个字段中获取突出显…

python:霍夫变换检测直线

霍夫变换检测直线 在Python中&#xff0c;可以使用OpenCV库来实现霍夫变换进行直线检测。 一、原理 1、霍夫变换(Hough Transform) 霍夫变换是图像处理中从图像中识别几何形状的基本方法之一&#xff0c;应用很广泛&#xff0c;也有很多改进算法。主要用来从图像中分离出具有…

Linux中的锁

user2正在进行抢票: 4 user3正在进行抢票: 3 user1正在进行抢票: 2 user4正在进行抢票: 1 user2正在进行抢票: 0 user3正在进行抢票: -1 user1正在进行抢票: -2 int tickets10000; void* getTicket(void* args) {string usernamestatic_cast<const char*>(args);while(…

【C++篇】迈入新世界的大门——初识C++(上篇)

文章目录 C发展历史C起源C版本更新C23小故事 C在工作领域的应用C参考网站及文档书籍编程语言排行榜C难度参考文档书籍参考文档参考书籍 C第一个程序命名空间为什么要使用namespacenamespace定义及规则命名空间使用 C输入&输出名字含义 缺省参数函数重载 C发展历史 C起源 …

新手小白零基础,该怎样学习编程呢?零基础入门到精通,收藏这一篇就够了

零基础编程入门先学什么&#xff1f;编程语言有几百种&#xff0c;我们应该怎么选择。想学习编程&#xff0c;加入互联网行业&#xff0c;哪一个更有前途&#xff1f;在小白学习编程会有各种各样的问题&#xff0c;今天小编我就来为你解答。 一、怎么选择编程语言 编程语言有很…

geomagic怎么删除平面?geomagic怎么修模

在现代三维建模和3D打印技术的发展中&#xff0c;Geomagic作为一款专业的软件工具&#xff0c;广泛应用于逆向工程、产品设计和质量检测等领域。本文将详细介绍geomagic怎么删除平面&#xff1f;geomagic怎么修模&#xff0c;并探讨Geomagic的主要应用领域。通过这些内容&#…

SAP_ABAP模块-批量导入货源清单

一、业务背景 有个朋友做ECC 6.0的项目&#xff0c;期初上线时&#xff0c;有一个需求是批量导入货源清单&#xff0c;我问了好几个朋友&#xff0c;加上自己以前的积累&#xff0c;硬是没有找到一个完全能用的程序&#xff0c;下面我来说一下我遇到的问题&#xff1b; 对货源清…

【软件造价咨询】软件造价之全国各省市功能点单价分析

在软件工程领域&#xff0c;功能点是衡量软件规模的一种单位&#xff0c;功能点分析是一种广泛使用的方法&#xff0c;用于估算软件项目的规模和成本。其中功能点单价是指每功能点的软件开发费用&#xff08;单位&#xff1a;元/功能点&#xff09;。 本篇文章通过调研了20多份…

运维开发——局域网SSH访问服务器与应用

摘要 本博文主要介绍局域网SSH访问登陆虚拟机和及其应用相关配置操作。 1. 局域网SSH访问登陆虚拟机 目标&#xff1a;在局域网内A电脑使用SSH登陆B电脑上虚拟机的服务器。 前提条件:B电脑为宿主机&#xff0c;可以正常使用ssh访问虚拟机服务器&#xff0c;虚拟机网络连接方…

【面试题】文本左右对齐

文本左右对齐 学习 一、题目 这个问题是一个典型的文本排版问题。 二、解题思路 初始化&#xff1a;创建一个结果列表result来存储每一行的文本&#xff0c;以及一个临时列表current_line来存储当前正在构建的行的单词。 贪心算法填充&#xff1a;遍历words数组&#xff0c;…

Linux:开发工具(2)

一、Linux编译器-gcc/g使用 1.1 为什么我们可以用C/C做开发呢&#xff1f; 无论是在windows、还是Linux中&#xff0c;C的开发环境不仅仅指的是vs、gcc、g&#xff0c;更重要的是语言本身的头文件&#xff08;函数的声明&#xff09;和库文件&#xff08;函数的实现&#xff0…

WPF动画

补间动画&#xff1a;动画本质就是在一个时间段内对象尺寸、位移、旋转角度、缩放、颜色、透明度等属性值的连续变化。也包括图形变形的属性。时间、变化的对象、变化的值 工业应用场景&#xff1a;蚂蚁线、旋转、高度变化、指针偏移、小车 WPF动画与分类 特定对象处理动画过…

本地项目上传github

一、先在github&#xff08;GitHub: Let’s build from here GitHub&#xff09;上创建仓库 1&#xff0c;登录github后&#xff0c;点击右上角头像&#xff0c;点击 Your repositories 2&#xff0c;点击new 3&#xff0c;填写仓库名&#xff0c;假设命名 testhub&#xff0…

【机器学习】全景指南:从基础概念到实战流程的全面解析

文章目录 1.引言1.1机器学习的重要性1.2机器学习的应用范围1.3本文的内容结构 2. 机器学习的基本概念与分类2.1 机器学习的定义2.2 机器学习的分类 4. 强化学习&#xff08;Reinforcement Learning&#xff09; 3. 机器学习的工作流程3.1 数据收集与准备1. 数据源与类型2. 数据…

win10怎么查看CPU多核占用率

想要看自己有几个CPU处理器&#xff0c;可以在设备管理器里查看&#xff1a; 查看多核占用率&#xff0c;搜索任务管理器&#xff0c;然后打开&#xff0c;任务管理器——性能——CPU 右下角就可以看到我的是1个CPU&#xff0c;6个内核&#xff0c;12线程 想要看每个CPU占用…

Unity3D 自定义窗口

Unity3D 自定义窗口的实现。 自定义窗口 Unity3D 可以通过编写代码&#xff0c;扩展编辑器的菜单栏和窗口。 简单的功能可以直接一个菜单按钮实现&#xff0c;复杂的功能就需要绘制一个窗口展示更多的信息。 编辑器扩展的脚本&#xff0c;需要放在 Editor 文件夹中。 菜单栏…

用Python爬取高德地图路径规划数据——01. 指定起终点爬取-Python程序及详解

这个Python程序旨在从高德地图API获取路径规划数据&#xff0c;解析这些数据&#xff0c;并最终将其保存到JSON和CSV文件中。下面&#xff0c;我将详细讲解每个部分的功能和实现方式。 1. 导入所需的模块 import requests import json import time import csvrequests: 用于发…

spring boot自动配置

Spring自动配置是Spring框架的一个核心特性&#xff0c;它允许开发者通过在类路径下的配置类发现bean&#xff0c;而无需在应用程序中显式地进行bean的声明。Spring Boot利用这一特性&#xff0c;通过starter依赖的机制和EnableAutoConfiguration注解&#xff0c;帮助开发者快速…

Adobe After Effects AE V2023-23.6.6.2 解锁版下载安装教程 (视频合成和特效制作)

前言 Adobe After Effects&#xff08;简称AE&#xff09;是一款专业的图形视频处理软件&#xff0c;数字影视特效合成软件&#xff0c;视频后期特效制作软件。主要用来创建动态图形和视觉特效&#xff0c;支持2D以及3D&#xff0c;是基于非线性编辑的软件&#xff0c;透过图层…

原生js用Export2Excel导出excel单级表头和多级表头数据方式实现

原生js用Export2Excel导出excel单级表头和多级表头数据方式实现 原生js用Export2Excel导出excel单级表头和多级表头数据方式实现HTML文件导入需要的文件HTML文件中实现导出函数HTML总代码实现汇总&#xff08;直接复制代码&#xff0c;注意js引入路径&#xff09; 原生js用Expo…