STM32驱动步进电机(原理、程序、解决电机只震动不转动问题)

2023-11-10

一、步进电机的介绍

首先来看一下步进电机的样子,本介绍采用平时最常见也是最简单的28BYJ-48,这是一个五线四项电机。

五线:顾名思义 外部五条线     

四项:电机内部的定子上有8个齿,正对着的2个齿上的绕组又是串联在一起的,也就是说正对着的2个绕组总是会同时导通或关断的,如此就形成了4相

 二、步进电机的原理及工作方式

原理:

步进电机是一种将电脉冲信号转换成相应角位移线位移的电动机。每输入一个脉冲信号转子就转动一个角度或前进一步,其输出的角位移线位移与输入的脉冲数成正比,转速脉冲频率成正比。

通俗易懂的说:就是给不同的项轮流通电,在电磁感应的作用下,每次产生一个很小的角位移,连贯起来就带动了电机的转动,从而驱动负载。

工作方式:

四拍驱动、八拍驱动。

四拍驱动:这是最简单的步进电机驱动方式。这种方式,电机在每个瞬间只有一个线圈导通。

                 A→B→C→D

第一个图更方便大家理解,第二个图就是相应拉高拉低。

大白话就是:按顺序给相应引脚拉高(1)拉低(0)。   一个拉高,其他三个拉低。

八拍驱动:就是在上述四拍切换过程中穿插两个线圈同时拉高的情况。

                 A→AB→B→BC→C→CD→D→DA

三、电机驱动板

由于单片机IO口输出电流过小,无法带动电机运行,因此我们需要另外加一个驱动板,用ULN2003就行,简单便宜。

 四、代码驱动

 话不多说直接上代码。

step_motor.h

#ifndef __STEP_MOTOR_H
#define __STEP_MOTOR_H	 
#include "sys.h"
#include "delay.h"

extern u8 STEP; //定义单步计数 全局变量


#define STEP_MOTOR_PORT			GPIOG	//定义IO接口所在组
#define STEP_MOTOR_A				GPIO_Pin_2	//定义IO接口
#define STEP_MOTOR_B				GPIO_Pin_3	//定义IO接口
#define STEP_MOTOR_C				GPIO_Pin_4	//定义IO接口
#define STEP_MOTOR_D				GPIO_Pin_5	//定义IO接口
#define STEP_MOTOR_CLK      RCC_APB2Periph_GPIOG



void STEP_MOTOR_Init(void);//初始化
void STEP_MOTOR_OFF (void);//断电状态
void STEP_MOTOR_8A (u8 a,u16 speed);
void STEP_MOTOR_NUM (u8 RL,u16 num,u8 speed);//电机按步数运行
void STEP_MOTOR_LOOP (u8 RL,u8 LOOP,u8 speed);//电机按圈数运行

#endif

step_motor.c

#include "step_motor.h"

u8 STEP; 


void STEP_MOTOR_Init(void){ //接口初始化
	
	GPIO_InitTypeDef GPIO_InitStruct;
	RCC_APB2PeriphClockCmd(STEP_MOTOR_CLK, ENABLE);

    GPIO_InitStruct.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStruct.GPIO_Pin = STEP_MOTOR_A;
    GPIO_Init(STEP_MOTOR_PORT, &GPIO_InitStruct);
    GPIO_InitStruct.GPIO_Pin = STEP_MOTOR_B;
    GPIO_Init(STEP_MOTOR_PORT, &GPIO_InitStruct);
    GPIO_InitStruct.GPIO_Pin = STEP_MOTOR_C;
    GPIO_Init(STEP_MOTOR_PORT, &GPIO_InitStruct);
    GPIO_InitStruct.GPIO_Pin = STEP_MOTOR_D;
    GPIO_Init(STEP_MOTOR_PORT, &GPIO_InitStruct);

    GPIO_ResetBits(STEP_MOTOR_PORT, STEP_MOTOR_A);
    GPIO_ResetBits(STEP_MOTOR_PORT, STEP_MOTOR_B);
    GPIO_ResetBits(STEP_MOTOR_PORT, STEP_MOTOR_C);
    GPIO_ResetBits(STEP_MOTOR_PORT, STEP_MOTOR_D);	
	
}

void STEP_MOTOR_OFF (void)  //电机断电 全拉低
{
GPIO_ResetBits(STEP_MOTOR_PORT,STEP_MOTOR_A | STEP_MOTOR_B | STEP_MOTOR_C | STEP_MOTOR_D);//各接口置0
}

void STEP_MOTOR_8A (u8 a,u16 speed)  //电机单步8拍
{
	switch (a){
		case 0:
		GPIO_ResetBits(STEP_MOTOR_PORT,STEP_MOTOR_B | STEP_MOTOR_C | STEP_MOTOR_D);//0
		GPIO_SetBits(STEP_MOTOR_PORT,STEP_MOTOR_A);//1
			break;
		case 1:
		GPIO_ResetBits(STEP_MOTOR_PORT,STEP_MOTOR_C | STEP_MOTOR_D);//0
		GPIO_SetBits(STEP_MOTOR_PORT,STEP_MOTOR_A | STEP_MOTOR_B);//1
			break;
		case 2:
		GPIO_ResetBits(STEP_MOTOR_PORT,STEP_MOTOR_A | STEP_MOTOR_C | STEP_MOTOR_D);//0
		GPIO_SetBits(STEP_MOTOR_PORT,STEP_MOTOR_B);//1
			break;
		case 3:
		GPIO_ResetBits(STEP_MOTOR_PORT,STEP_MOTOR_A | STEP_MOTOR_D);//0
		GPIO_SetBits(STEP_MOTOR_PORT,STEP_MOTOR_B | STEP_MOTOR_C);//1
			break;
		case 4:
		GPIO_ResetBits(STEP_MOTOR_PORT,STEP_MOTOR_A | STEP_MOTOR_B | STEP_MOTOR_D);//0
		GPIO_SetBits(STEP_MOTOR_PORT,STEP_MOTOR_C);//1
			break;
		case 5:
		GPIO_ResetBits(STEP_MOTOR_PORT,STEP_MOTOR_A | STEP_MOTOR_B);//0
		GPIO_SetBits(STEP_MOTOR_PORT,STEP_MOTOR_C | STEP_MOTOR_D);//1
			break;
		case 6:
		GPIO_ResetBits(STEP_MOTOR_PORT,STEP_MOTOR_A | STEP_MOTOR_B | STEP_MOTOR_C);//0
		GPIO_SetBits(STEP_MOTOR_PORT,STEP_MOTOR_D);//1
			break;
		case 7:
		GPIO_ResetBits(STEP_MOTOR_PORT,STEP_MOTOR_B | STEP_MOTOR_C);//0
		GPIO_SetBits(STEP_MOTOR_PORT,STEP_MOTOR_A | STEP_MOTOR_D);//1
			break;
		default:
			break;
	}
	delay_ms(speed); //延时
	STEP_MOTOR_OFF();//进入断电状态,防电机过热
}


void STEP_MOTOR_NUM (u8 RL,u16 num,u8 speed)  //电机按步数运行
{
	u16 i;
	for(i=0;i<num;i++)
    {	
		if(RL==1){ //当RL=1右转,RL=0左转
			STEP++;
			if(STEP>7)STEP=0;
		}else{
			if(STEP==0)STEP=8;
			STEP--;
		}
		STEP_MOTOR_8A(STEP,speed);
	}
}

void STEP_MOTOR_LOOP (u8 RL,u8 LOOP,u8 speed)  //电机按圈数运行
{
	STEP_MOTOR_NUM(RL,LOOP*4076,speed); 
}

主函数 main.c

在主函数中调用这个函数来驱动电机。

函数第一个参数:控制正反转 1→正转   0→反转
    第二个参数:控制电机转的圈数
    第三个参数:延时

STEP_MOTOR_LOOP(1,1,1);  //步进电机正传

STEP_MOTOR_LOOP(0,1,1);  //步进电机反传


五、电机只振东不转动问题

①电机只震动,却不转动(正反都不转动)

可以适当调节电机驱动函数的延时时间,就是调节8拍切换的延时,调大(2s)、调小(1s)都试一下,如果不是电机损坏的话  这种方法都可以解决问题。

 ②还有一种情况,电机正转可以转动、反转只震动;或者正转只震动、反转正常

这种情况会发现演示调为1s、2s都不好使,总是有一个正常、一个不正常,,,这时候需要将延时取中间大小,比如说1.5s,,,但是呢,你会发现程序中的 delay_ms() 函数只能设置整数,要么1 要么2,不能设置1.5,,这就需要我们更改一下  delay_ms()  函数了。改成这样即可:

 

全部工程链接:https://item.taobao.com/item.htm?spm=a1z10.1-c.w4004-24450682672.26.7aa61b42erNczw&id=693670548640

                                   欢迎大家指正交流,有空可以一起讨论代码啊。

                                                                                                             --------------一个正在努力的人

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

STM32驱动步进电机(原理、程序、解决电机只震动不转动问题) 的相关文章

  • C#文件下载四种方法

    从CSDN论坛上看到这个帖子 http topic csdn net u 20120822 10 d8115bb7 2f2a 4c2f b0c1 aab68bcb1e3e html 出于对C 功能的好奇 想那个工程试看看 一 过程 1 建C
  • idea插件--java类实体vo转json串(POJO to JSON)

    安装POJO to JSON插件 1 打开idea 右键选择File setting Plugins 输入POJO to JSON 选择安装后重启idea Invalidate and Restart vo转换JSON操作 Data pub
  • 连续型随机变量的分布(均匀分布、指数分布、正态分布)

    一 均匀分布 均匀分布是指在一个区间内各个数值出现的概率相等的一种随机分布 均匀 这一概念可以理解为 在任何子区间上 变量的取值概率相等 它的概率密度函数为 其中 a和b分别为区间的上下限 均匀分布的特点是 它的概率密度函数为常数 即该分布

随机推荐