STM32控制步进电机:基于定时器中断的ULN2003驱动器/步进电机驱动程序

修改

2023.04.24 修改内容:完善了下程序,添加了工作原理中的一些内容。
看了评论区一位朋友指出的问题,然后我对文章中的工作原理、和程序做了些小修改(修改位置为:一、1 2 ;三、1 ;五),方便之前收藏了的朋友查阅,程序源文件也重新上传了。该篇文章中的一圈所需脉冲计算为4096个脉冲,但是其实我个人测试时为2048个脉冲,驱动器输出用了示波器测了也都是正常的(一、2),目前还不清楚原因。为了不误导其他后来的朋友,先在这里说明一下。如果有大佬知道是什么原因欢迎批评指正。

一、ULN2003驱动器

1、工作原理

下图为ULN2003驱动器原理图。
在这里插入图片描述
此驱动器的原理即为步进电机的工作原理,此篇文章有介绍到:STM32控制步进电机:工作原理及库函数(标准库) / HAL库控制程序(不定期更新)
通过一个接一个的引脚驱动电机的4个相,使得步进电机转动,该驱动器输入为低电平时,对应输出引脚输出为高电平,反之,输入为高电平时,输出为低电平。因此要使得某些引脚输出为高电平时,应当置其对应的引脚为低电平,其他引脚为高电平。

2、步距角以及一圈所需步数的计算

本篇文章使用的4相5线步进电机步距角为5.625/64,因此步进电机转一圈需要走360°/步距角的步数,即(360/5.625)*64 = 4096步
虽然计算是一圈4096个脉冲,但是实际上我使用的时候是2048个脉冲电机转一圈,于是我使用了示波器测试了程序是否有问题。
如下所示白圈中所示:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

二、硬件连接

如图所示,记得共地。4个控制引脚在下方程序的.h和.c文件中有定义和引用到。
在这里插入图片描述

三、STM32F103定时器中断控制步进电机程序

此程序效果为正转1周后再反转1周回到原点。程序如下所示。

1、.c文件

以下为步进电机驱动的motor.c文件

#include "motor.h"

//num用于对引脚的索引,j用于计算步数,fx为电机旋转方向
unsigned short int num=0,j,fx;

void motor_GPIO_Init(void)
{
   
 
 GPIO_InitTypeDef  GPIO_InitStructure;
 	
 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);	 //使能PB端口时钟
	
 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5 | GPIO_Pin_6 | GPIO_Pin_7 | GPIO_Pin_8;
 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; 		 //推挽输出
 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;		 //IO口速度为50MHz
 GPIO_Init(GPIOB, &GPIO_InitStructure);					 //根据设定参数初始化GPIOB
 GPIO_ResetBits(GPIOB,GPIO_Pin_5 | GPIO_Pin_6 | GPIO_Pin_7 | GPIO_Pin_8);						 //PB.5/6/7/8 输出低电平
}

void TIM3_Int_Init(u16 arr,u16 psc)
{
   
  TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
	NVIC_InitTypeDef NVIC_InitStructure;
以下是一个简单的示例程序,使用STM32F10C8T6的定时器控制ULN2003步进电机。这个程序可以控制电机顺时针和逆时针旋转。 ```c #include "stm32f10x.h" void delay_ms(uint32_t ms); int main(void) { // 初始化GPIO RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); GPIO_InitTypeDef GPIO_InitStruct; GPIO_InitStruct.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3; GPIO_InitStruct.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOA, &GPIO_InitStruct); // 初始化定时器 RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); TIM_TimeBaseInitTypeDef TIM_InitStruct; TIM_InitStruct.TIM_Period = 1000 - 1; // 计数器周期 TIM_InitStruct.TIM_Prescaler = 7200 - 1; // 预分频器 TIM_InitStruct.TIM_ClockDivision = TIM_CKD_DIV1; TIM_InitStruct.TIM_CounterMode = TIM_CounterMode_Up; TIM_InitStruct.TIM_RepetitionCounter = 0; TIM_TimeBaseInit(TIM2, &TIM_InitStruct); // 启动定时器 TIM_Cmd(TIM2, ENABLE); int step_count = 0; while (1) { if (step_count < 512) { GPIO_SetBits(GPIOA, GPIO_Pin_0); delay_ms(1); GPIO_ResetBits(GPIOA, GPIO_Pin_0); delay_ms(1); step_count++; } else if (step_count < 1024) { GPIO_SetBits(GPIOA, GPIO_Pin_1); delay_ms(1); GPIO_ResetBits(GPIOA, GPIO_Pin_1); delay_ms(1); step_count++; } else if (step_count < 1536) { GPIO_SetBits(GPIOA, GPIO_Pin_2); delay_ms(1); GPIO_ResetBits(GPIOA, GPIO_Pin_2); delay_ms(1); step_count++; } else if (step_count < 2048) { GPIO_SetBits(GPIOA, GPIO_Pin_3); delay_ms(1); GPIO_ResetBits(GPIOA, GPIO_Pin_3); delay_ms(1); step_count++; } else { step_count = 0; } } } void delay_ms(uint32_t ms) { while (ms--) { volatile uint32_t i = 72000; while (i--) { __NOP(); } } } ``` 这个程序使用TIM2定时器和GPIOA的四个引脚控制步进电机。使用定时器产生1ms的延迟来控制步进电机的旋转速度。注意,这个程序中的延迟函数并不精确,实际使用时需要根据具体的系统时钟和定时器配置来调整延迟时间。
评论 23
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

超级大魔王li

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值