51单片机怎么驱动直流电机c语言

51单片机怎么驱动直流电机c语言,第1张

51单片机驱动直流电机程序(用的是l298n芯片):

#include<reg51.h>

#include<math.h>

#defineuintunsignedint

#defineucharunsignedchar

#defineN100

sbit s1=P1^0//电机驱动口

sbits2=P1^1//电机驱动口

sbits3=P1^2//电机驱动口

sbits4=P1^3//电机驱动口

sbiten1=P1^4//电机使能端

sbiten2=P1^5//电机使能端

sbitLSEN=P2^0//光电对管最左

sbitLSEN1=P2^1//光电对管左1

sbitLSEN2=P2^2//光电对管左2

sbitRSEN1=P2^3//光电对管右1

sbitRSEN2=P2^4//光电对管右2

sbitRSEN=P2^5//光电对管最右

uintpwm1=0,pwm2=0,t=0

voiddelay(uintxms)

{

uinta

while(--xms)

{

for(a=123a>0a--)

}

}

voidmotor(ucharspeed1,ucharspeed2)

{

if(speed1>=-100&&speed1<=100)

{

pwm1=abs(speed1)

if(speed1>0)

{

s1=1

s2=0

}

if(speed1==0)

{

s1=1

s2=1

}

if(speed1<0)

{

s1=0

s2=1

}

}

if(speed2>=-100&&speed2<=100)

{

pwm2=abs(speed2)

if(speed2>0)

{

s3=1

s4=0

}

if(speed2==0)

{

s3=1

s4=1

}

if(speed2<0)

{

s3=0

s4=1

}

}

}

voidgo_forward(uintspeed)

{

s1=1

s2=0

s3=1

s4=0

pwm1=speed

pwm2=speed

}

voidgo_back(uintspeed)

{

s1=0

s2=1

s3=0

s4=1

pwm1=speed

pwm2=speed

}

voidstop()

{

s1=1

s2=1

s3=1

s4=1

pwm1=0

pwm2=0

}

voidturn_right(uintP1,uintP2)//右转函数

{

s1=1

s2=0

s3=0

s4=1

pwm1=P1

pwm2=P2

}

voidturn_left(uintP1,uintP2)//左转函数

{

s1=0

s2=1

s3=1

s4=0

pwm1=P1

pwm2=P2

}

voidtracking()

{

if((LSEN1==0)&&(LSEN2==0)&&(RSEN1==0)&&(RSEN2==0))//没有检测到

{

go_forward(100)

}

if((LSEN1==1)&&(LSEN2==0)&&(RSEN1==0)&&(RSEN2==0))//左一检测到

{

turn_left(40,80)//左转右轮》左轮

delay(N)

}

if((LSEN1==0)&&(LSEN2==1)&&(RSEN1==0)&&(RSEN2==0))//左二检测到

{

turn_left(40,60)//左转右轮》左轮

delay(N)

}

if((LSEN1==0)&&(LSEN2==0)&&(RSEN1==1)&&(RSEN2==0))//右一检测到

{

turn_right(60,4)//右转左轮》右轮

delay(N)

}

if((LSEN1==0)&&(LSEN2==0)&&(RSEN1==0)&&(RSEN2==1))//右二检测到

{

turn_right(80,40)//右转左轮》右轮

delay(N)

}

if((LSEN1==1)&&(LSEN2==1))

{

turn_left(0,100)

delay(1000)

}

if((RSEN1==1)&&(RSEN2==1))

{

turn_right(100,0)

delay(1000)

}

}

voidavoidance()

{

}

voidinit()

{

TMOD=0x02//timer0同时配置为模式2,8自动重装计数模式

TH0=156//定时器初值设置100us中断

TL0=156

ET0=1

EA=1

TR0=1//开启总中断

}

voidmain()

{

init()

while(1)

{

tracking()

}

}

voidtimer0()interrupt1//电机驱动提供PWM信号

{

if(t<pwm1)

en1=1

else

en1=0

if(t<pwm2)

en2=1

else

en2=0

t++

if(t>100)

t=0

}

扩展资料

L298N是一种双H桥电机驱动芯片,其中每个H桥可以提供2A的电流,功率部分的供电电压范围是2.5-48v,逻辑部分5v供电,接受5vTTL电平。一般情况下,功率部分的电压应大于6V否则芯片可能不能正常工作。

参考资料来源:百度百科-l298n

单片机控制步进电机,我想你说的是两相步进电机,一般是控制其相序分配的顺逆从而控制正反转,一般而言,步进电机相序分配你可以做成一个数组比如step[]={0x03,0x06,0x0c,0x09},这样来说可以假设P0口是步进电机控制口,那么可以按如下方式来控制: while(1) { for(i=0i<4i++) { if(fx==1)P0=step[i]//正向 else P0=step[3-i]//反向 delay(x)//x大小决定电机速度。

根据电机相数买个驱动器。然后用单片机产生脉冲来控制电机的转动以及正反转。单片机产生脉冲的方法和单片机控制流水灯是一样的。ULN2003D 是驱动步进电机的驱动芯片,主要是匹配电机所需的电流。 由于是四相电机,步进电机之所以可以转动就需要给相绕组提供连续的脉冲,所以需要4个端口来控制四相绕组的工作状态(P15应该是不需要的),具体的编码要看电机的拍数; 一旦明白这些,你就可以很容易编写代码来控制电机的转动了,还有在脉冲间你可以设置不同的延时时间来调节电机的转速。

#include<reg51.h>

#define uchar unsigned char

#define uint unsigned int

sbit PWM11 = P1^0 //motora

sbit PWM12 = P1^1

sbit PWM21 = P1^2 //motorb

sbit PWM22 = P1^3

sbit MC1 = P1^4 //motora enable

sbit MC2 = P1^5 //motorb enable

sbit LED = P1^6 //指示灯

sbit FMQ = P1^7 //蜂鸣器

sbit key_qt = P0^0 //motor2 引脚

sbit key_sph = P0^1 //升速键

sbit key_spl = P0^2 //降速键

sbit limita1 = P0^3 //电机正a限位

sbit limita2 = P0^4 //电机负a限位

sbit limitb1 = P0^5 //电机正b限位

sbit limitb2 = P0^6 //电机负b限位

uchar keyval,cnt=0,cnt1=0,pwm=50,num

uchar m_mode=0 //电机运行模式,1-电机正向;2-电机b正反向;3=电机b反向;4-电机a反向;

bit flag=0 //延时时间到标志

uchar runflag=0 //电机运行标志

void delaynms(uint z)

{

uint i,j

for(i = zi >0i--)

for(j = 110j >0j--)

}

void keychk(void)

{

keyval=0xff

if(key_qt==0)

{

delaynms(5)//消抖

if(key_qt==0) //确定按键按下

{

while(key_qt==0)

keyval=1

}

}

if(!key_sph)

{

delaynms(5)//消抖

if(!key_sph) //确定按键按下

{

while(!key_sph)

keyval=2

}

}

if(!key_spl)

{

delaynms(5)//消抖

if(!key_spl) //确定按键按下

{

while(!key_spl)

keyval=3

}

}

}

void t0isr() interrupt 1

{

TH0=(65536-50000)/256

TL0=(65536-50000)%256

if(runflag==1) //正常运行

{

cnt1++

if(cnt1>=10){cnt1=0LED=~LED}

}

if(runflag==2) //限位延时

{

LED=1

cnt++

if(cnt>=10)

{

flag=1

cnt=0

}

}

}

void t1isr() interrupt 3

{

num++

if(num<pwm)

{

switch(m_mode)

{

case 1:PWM11=1PWM12=0break //a电机正转

case 2:PWM21=1PWM22=0break //b电机正转

case 3:PWM21=0PWM22=1break //b电机反转

case 4:PWM11=0PWM12=1break //a电机反转

default:break

}

}

else if(num>=pwm)

{

PWM11=0PWM12=0PWM21=0PWM22=0

}

if(num>=100)num=0

}

void main(void)

{

bit qt=0

TMOD=0x21

TH0=(65536-50000)/256

TL0=(65536-50000)%256

TH1=236 //改变该值调整PWM频率

TL1=236

TR0=1

TR1=1

ET1=1

ET0=1

EA=1

P1|=0xc0

flag=0

while(1)

{

keychk()

switch(keyval)

{

case 1: qt=~qt

if(qt)

{

runflag=1

switch(m_mode)

{

case 0: m_mode=1 //刚开始启动,置模式1

MC1=1

FMQ=0 //蜂鸣器响一下

delaynms(100)

FMQ=1

break

case 1:MC1=1break //电机a正转

case 2:MC2=1break //电机b正转

case 3:MC2=1break //电机b反转

case 4:MC1=1break //电机a反转

default:m_mode=0break

}

}

if(!qt)

{

MC1=0

MC2=0

LED=1

runflag=0

break

}

break

case 2:pwm+=5if(pwm>100)pwm=100break //升速5个单位

case 3:pwm-=5if(pwm<10)pwm=10break //降速5个单位

default:break

}

if(limita1==0 &&m_mode==1) //电机a正向限位

{

delaynms(1)

if(limita1==0)

{

MC1=0

runflag=2

flag=0

while(flag==0) //等待延时0.5S结束

m_mode=2

runflag=1

MC2=1

}

}

if(limitb1==0 &&m_mode==2) //电机b正向限位

{

delaynms(1)

if(limitb1==0)

{

MC2=0

runflag=2

flag=0

while(flag==0)

qt=0

m_mode=3

}

}

if(limitb2==0 &&m_mode==3) //电机b反向限位

{

delaynms(1)

if(limitb2==0)

{

MC2=0

runflag=2

flag=0

while(flag==0)

m_mode=4

runflag=1

MC1=1

}

}

if(limita2==0 &&m_mode==4) //电机a反向限位

{

delaynms(1)

if(limita2==0)

{

MC1=0

runflag=2

flag=0

while(flag==0)

flag=0

m_mode=0

runflag=0

qt=0

LED=1

}

}

}

}


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/yw/7945339.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-04-11
下一篇 2023-04-11

发表评论

登录后才能评论

评论列表(0条)

保存