上周总算做成了点事,其实很简单,就是用Arduino实现了步进电机的正转和反转。因为总是很忙,所以拖到现在才有时间补上攻略。感慨一下,IT民工真辛苦啊,有位同事的昵称已经改成 “还是古代好,切掉小JJ就可以当公务员了”。 好吧,祝这位朋友心想事成,我还是坚持当民工好了
在步进电机实验前,从网上找到了一个关于步进电机驱动的文档,使用的是L298N的驱动芯片。这个文档里面有些错误,导致我浪费了大量时间。也许不同的电机和芯片版本也许有区别吧,请大家慎重使用。先借用一下原理图:
错误的L298N步进电机驱动芯片原理图
其中IN1~IN4以及ENA,ENB在之前的一篇关于直流电机驱动中有介绍过,分别对应输入的六个管脚。这个图里有两处错误,首先可以看到有两个ENA,其中一个应该是ENB;另外这两个EN都是接地的,但是经我实验看来,应该接+5V的电压才对。
所以正确的原理图应该是这样的:
正确的L298N原理图
下面是对应的接线图:
乱七八糟的接线图
接线之后按照说明书里的时序图写了一段程序,呼呼,一次就通过了电机的正转实验。可是在企图实现反转的时候,总是不成功。在焦头烂额之后,终于开始怀疑文档是否正确。事实证明,大家做任何事情都要相信伟大的党而不是万恶的“文档”,下面是错误的反转时序图和经我验证可行的时序图:
步进电机驱动的时序图
基于以上时序图,我写了一段代码,让步进电机实现下面的动作:
正转一圈 -> 暂停1秒钟 -> 反转一圈 -> 暂停一秒钟 ->循环
代码如下:
int LeftI1 = 28; //连接电机驱动板的I1接口
int LeftI2 = 22; //连接电机驱动板的I2接口
int LeftEA = 8; //连接电机驱动板的EA接口
int RightI1 = 36; //连接电机驱动板的I1接口
int RightI2 = 42; //连接电机驱动板的I2接口
int RightEB = 6; //连接电机驱动板的EB接口
int StepCount = 0;
int StepDelayTime=1500;
void setup()
{
pinMode(LeftI1, OUTPUT); //I1和I2都是数字信号
pinMode(LeftI2, OUTPUT); //通过设置I1和I2来控制电机旋转方向
pinMode(LeftEA, OUTPUT); //按占空比方式输出的模拟信号
pinMode(RightI1, OUTPUT); //I1和I2都是数字信号
pinMode(RightI2, OUTPUT); //通过设置I1和I2来控制电机旋转方向
pinMode(RightEB, OUTPUT); //按占空比方式输出的模拟信号
Serial.begin(9600); //设置波特率
}
void ForwardInit()
{
digitalWrite(LeftEA, HIGH);
digitalWrite(RightEB,HIGH );
digitalWrite(LeftI1, LOW);
digitalWrite(LeftI2,HIGH );
digitalWrite(RightI1,HIGH);
digitalWrite(RightI2, HIGH);
StepCount=0;
}
void BackwardInit()
{
digitalWrite(LeftEA, HIGH);
digitalWrite(RightEB,HIGH );
digitalWrite(LeftI1, LOW);
digitalWrite(LeftI2,LOW );
digitalWrite(RightI1,LOW);
digitalWrite(RightI2, HIGH);
StepCount=0;
}
void ForwardOneStep()
{
delayMicroseconds(StepDelayTime);
switch(StepCount)
{
case 0:
digitalWrite(RightI2,LOW);
digitalWrite(LeftI1,HIGH);
break;
case 1:
digitalWrite(RightI1,LOW);
digitalWrite(RightI2,HIGH);
break;
case 2:
digitalWrite(LeftI2,LOW);
digitalWrite(RightI1,HIGH);
break;
case 3:
digitalWrite(LeftI1,LOW);
digitalWrite(LeftI2,HIGH);
break;
}
StepCount=(StepCount + 1) % 4;
}
void BackwardOneStep()
{
delayMicroseconds(StepDelayTime);
switch(StepCount)
{
case 0:
digitalWrite(RightI2,LOW);
digitalWrite(LeftI1,HIGH);
break;
case 1:
digitalWrite(LeftI1,LOW);
digitalWrite(LeftI2,HIGH);
break;
case 2:
digitalWrite(LeftI2,LOW);
digitalWrite(RightI1,HIGH);
break;
case 3:
digitalWrite(RightI1,LOW);
digitalWrite(RightI2,HIGH);
break;
}
StepCount=(StepCount + 1) % 4;
}
void loop()
{
while(1)
{
ForwardInit();
for(int i=0;i<200;i++)
{
ForwardOneStep();
}
delay(1000);
BackwardInit();
for(int i=0;i<200;i++)
{
BackwardOneStep();
}
delay(1000);
}
}
除了实现了动作之外,我还搭车实验了下面几件事情:
1,步进电机的转角相当精确,我捆了根电线当指针,反复转了几百圈之后,指针的位置几乎没有变化
2,扭矩还挺大,我选用的是标称扭矩是3.4Kg.cm的步进电机,用铅酸蓄电池供电,旋转时我用爪子完全不能把它捏住(NXT的电机貌似没有这么强劲)。
3,经我测试,每个脉冲之间的间距最好大于1500μs(1.5ms),如果间距太小的话,就会出现失步的情况。
思考问题:基本上来说,每个步进电机都需要一个驱动板(L298N)和一个控制板(Arduino或其它单片机)。如果需要控制多个电机的话(小爱也许会有20多个关节),买这么多板子成本就太高了。实际上,每个脉冲间距之间有1500微秒的空闲时间,对CPU来说简直是漫漫长夜。所以我觉得可以用类似于操作系统多任务的思想来生成时序,充分利用脉冲间距之间的剩余价值,这样就可以只用一块Arduino实验板来控制多个电机了。
呵呵,又想多了,等下周有空再试试吧!