STM32CubeMX Hal库 步进电机驱动 [ULN2003驱动器] 阻塞模式和非阻塞模式[中断模式]

吐槽一下,今天准备使用ULN2003来驱动步进电机,我用的stm32cubemx ,但是呢csdn上居然找不到我满意的(写的五花八门),emmm,没办法自己写一个吧。


前言

使用的是ULN2003驱动器,这个驱动起来很简单,只要给四个引脚不同顺序的高电平就能使步进电机转动,本文分为阻塞式和非阻塞式,也就是一个必须在主循环中一直等着,这样会影响其他程序的运行,一种是非阻塞式,我们把他放在中断里面跑,这样就能不影响主程序运行的情况下运行电机了。
在这里插入图片描述


一、阻塞模式

阻塞模式很简单,只要在主循环里面进行循环即可

1.1 cubemx配置

在这里插入图片描述
在cubemx里面只需要设置这四个引脚就可以了

1.2 代码编写

bujin.c文件

#include "bujin.h"

void stepper(uint8_t dir,int speed)
{
	if(dir == Pos)
	{
		HAL_GPIO_WritePin(GPIOG,GPIO_PIN_6,GPIO_PIN_SET);
		HAL_Delay(speed);
		HAL_GPIO_WritePin(GPIOG,GPIO_PIN_6,GPIO_PIN_RESET);
		HAL_GPIO_WritePin(GPIOG,GPIO_PIN_7,GPIO_PIN_SET);
		HAL_Delay(speed);
		HAL_GPIO_WritePin(GPIOG,GPIO_PIN_7,GPIO_PIN_RESET);
		HAL_GPIO_WritePin(GPIOG,GPIO_PIN_8,GPIO_PIN_SET);
		HAL_Delay(speed);
		HAL_GPIO_WritePin(GPIOG,GPIO_PIN_8,GPIO_PIN_RESET);
		HAL_GPIO_WritePin(GPIOG,GPIO_PIN_9,GPIO_PIN_SET);
		HAL_Delay(speed);
		HAL_GPIO_WritePin(GPIOG,GPIO_PIN_9,GPIO_PIN_RESET);
	}
	else
	{
		HAL_GPIO_WritePin(GPIOG,GPIO_PIN_6,GPIO_PIN_SET);
		HAL_Delay(speed);
		HAL_GPIO_WritePin(GPIOG,GPIO_PIN_6,GPIO_PIN_RESET);
		HAL_GPIO_WritePin(GPIOG,GPIO_PIN_9,GPIO_PIN_SET);
		HAL_Delay(speed);
		HAL_GPIO_WritePin(GPIOG,GPIO_PIN_9,GPIO_PIN_RESET);
		HAL_GPIO_WritePin(GPIOG,GPIO_PIN_8,GPIO_PIN_SET);
		HAL_Delay(speed);
		HAL_GPIO_WritePin(GPIOG,GPIO_PIN_8,GPIO_PIN_RESET);
		HAL_GPIO_WritePin(GPIOG,GPIO_PIN_7,GPIO_PIN_SET);
		HAL_Delay(speed);
		HAL_GPIO_WritePin(GPIOG,GPIO_PIN_7,GPIO_PIN_RESET);
	}
}

bujin.h

在bujin.h中写
#ifndef __BUJIN_H
#define __BUJIN_H
#include "main.h"
enum dir{Pos,Neg};
void stepper(uint8_t dir,int speed);
#endif

很简单就是这样,就结束了
只要在主函数中调用就行了。
搞不懂为什么有的那个罗里吧嗦的。。。。。。
在这里插入图片描述
这里的5就是延时的时间,这个越小电机转的越快。反之就慢,但是不能小,当这个值过小的时候电机丢步!!!!
你要是用标准库的话就需要自己配置gpio口就行。

二、非阻塞式(中断模式)

中断模式,那什么来触发中断,由于控制的是电机,上面阻塞式的时候可以看到,每次设置完成以后都有一个延时,那么很好理解,要是在那段延时的时候我在敢其他事情(其他函数),当时间到了以后我再来驱动下一个gpio口就行了!

2.1 cubemx配置

我开启的是定时器6,我只要勾选activated就行,然后你把预分配和arr的值设置成我这样就行。
在这里插入图片描述
要是你的不是activated.那就应该是这个。
在这里插入图片描述
然后=设置完成了,还需要开启中断哈。
在这里插入图片描述

2.2 代码编写

bujin.c中写

#include "bujin.h"

int steep_dir__=0;

void stepper_(uint8_t dir)
{
	static int pos_=0;
	static int Neg_=0;
	if(dir == Pos)
	{
		if(pos_==0)
			HAL_GPIO_WritePin(GPIOG,GPIO_PIN_6,GPIO_PIN_SET);
		else if(pos_==1)
		{
			HAL_GPIO_WritePin(GPIOG,GPIO_PIN_6,GPIO_PIN_RESET);

			HAL_GPIO_WritePin(GPIOG,GPIO_PIN_7,GPIO_PIN_SET);
		}
		else if(pos_==2)
		{
			HAL_GPIO_WritePin(GPIOG,GPIO_PIN_7,GPIO_PIN_RESET);
			HAL_GPIO_WritePin(GPIOG,GPIO_PIN_8,GPIO_PIN_SET);
		}
		
		else if(pos_==3)
		{
			HAL_GPIO_WritePin(GPIOG,GPIO_PIN_8,GPIO_PIN_RESET);
			HAL_GPIO_WritePin(GPIOG,GPIO_PIN_9,GPIO_PIN_SET);
		}
		else if(pos_==4)
			HAL_GPIO_WritePin(GPIOG,GPIO_PIN_9,GPIO_PIN_RESET);
		
		pos_++;
		if(pos_==5)
		{
			pos_=0;
		}
	}
	else
	{
		if(Neg_==0)
		
			HAL_GPIO_WritePin(GPIOG,GPIO_PIN_6,GPIO_PIN_SET);
		
		else if(Neg_==1)
		{
			HAL_GPIO_WritePin(GPIOG,GPIO_PIN_6,GPIO_PIN_RESET);
			HAL_GPIO_WritePin(GPIOG,GPIO_PIN_9,GPIO_PIN_SET);
		}
		else if(Neg_==2)
		{
			HAL_GPIO_WritePin(GPIOG,GPIO_PIN_9,GPIO_PIN_RESET);
			HAL_GPIO_WritePin(GPIOG,GPIO_PIN_8,GPIO_PIN_SET);
		}
		else if(Neg_==3)
		{
			HAL_GPIO_WritePin(GPIOG,GPIO_PIN_8,GPIO_PIN_RESET);
			HAL_GPIO_WritePin(GPIOG,GPIO_PIN_7,GPIO_PIN_SET);
		}
		else if(Neg_==4)
			HAL_GPIO_WritePin(GPIOG,GPIO_PIN_7,GPIO_PIN_RESET);
		
		Neg_++;
		if(Neg_==5)
		{
			Neg_=0;
		}
	}
}

代码很烂emm但是能用,后面会改进一下,这里要注意,这个比较难完成快速的反复转动。要是介意的话不要用!

bujin.h中写

#ifndef __BUJIN_H
#define __BUJIN_H
#include "main.h"
extern int steep_dir__;
enum dir{Pos,Neg};
void stepper_(uint8_t dir);
#endif

it.c中写

void TIM6_IRQHandler(void)//要找到自己开的定时器中断时哪个!
{
  /* USER CODE BEGIN TIM6_IRQn 0 */
  /* USER CODE END TIM6_IRQn 0 */
  HAL_TIM_IRQHandler(&htim6);
  /* USER CODE BEGIN TIM6_IRQn 1 */
	stepper_(steep_dir__);//直接在这里调用就行
  /* USER CODE END TIM6_IRQn 1 */
}

主函数
这是控制方向的

steep_dir__=1;//逆时针
steep_dir__=0;//顺时针

这是控制速度的

	__HAL_TIM_SET_AUTORELOAD(&htim6,2000);//2000为快
	__HAL_TIM_SET_AUTORELOAD(&htim6,5000);//5000为慢

这是开启和关闭

	HAL_TIM_Base_Start_IT(&htim6);//这个是开启步进电机
	HAL_TIM_Base_Stop_IT(&htim6); //这个是关闭步进电机

这是一个简单实现正反转的代码。这些要在主循环之前调用

在这里插入图片描述

总结

代码仅限于能用,我是小白,大佬勿喷!!!

### 使用STM32CubeMX配置STM32控制步进电机 #### 准备工作 为了成功配置STM32控制步进电机,需准备好必要的硬件组件软件环境。所需材料包括但不限于STM32开发板(如STM32F103C8T6)、步进电机(例如28BYJ-48)、步进电机驱动模块(如ULN2003),以及面包板连接线等辅助设备[^2]。 #### 创建新项目并设置MCU参数 启动STM32CubeMX应用程序,在初始界面点击“New Project”,选择对应的STM32型号作为目标单片机。完成之后进入主窗口,这里可以对系统的各个外设进行详细的设定。 对于步进电机控制系统而言,主要关注的是GPIO引脚分配给步进电机的方向控制与脉冲输入;同时也要注意时钟树的合理规划以满足后续操作频率的需求[^1]。 #### 定义PWM信号源及时序参数 为了让STM32能够向步进电机发送精确的速度指令,通常会利用内置定时器产生的PWM波形来进行调制。因此,在此阶段应该挑选合适的通用定时器资源,并将其通道映射至特定的I/O端口上用于输出PWM信号。此外还需指定周期长度、占空比范围等相关属性以便于后期编程处理[^4]。 #### 初始化代码生成 当所有的硬件抽象层(HAL)函数都已正确关联后,就可以让STM32CubeMX自动生成初始化代码框架了。这一步骤不仅简化了底层寄存器的操作流程,同时也提供了丰富的API接口供开发者灵活运用。保存工程文件并导出为兼容IDE的工作空间形式,比如STM32CubeIDE所支持的标准格式。 ```c // 示例:初始化TIM2 PWM 输出 __HAL_RCC_TIM2_CLK_ENABLE(); htim2.Instance = TIM2; htim2.Init.Prescaler = 79; htim2.Init.CounterMode = TIM_COUNTERMODE_UP; htim2.Init.Period = 999; // 设置最大计数值 if (HAL_TIM_PWM_Init(&htim2) != HAL_OK){ Error_Handler(); } ``` #### 编写控制逻辑 最后就是根据具体的业务场景设计合理的运动轨迹算法,例如直线加速/减速过程中的速度变化规律计算等。这部分往往涉及到较为复杂的数学运算模型构建,但借助现成开源社区分享出来的成熟方案也可以大大缩短研发周期。
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值