【.NET 与树莓派】控制舵机
不管是小马达,还是大马达,嗯,也就是电机,相信大伙伴们也不会陌生。四驱车是一种很优秀的玩具,从老周小时候就开始流行(动画片《四驱兄弟》估计很多大朋友都看过),直到现在还能看到很多卖四驱车的。为啥会想起四驱车呢?因为小时候玩四驱车让老周认识了很多奇葩马达,什么“红魔鬼”、“蓝芯”、“紫魔鬼”……也不知道是什么逻辑的命名方式,反正那时候大家都这么叫。
这些马达的结构有一个转轴,绕有红色或橙色的线圈(记忆中是这些颜色),有四块磁铁。金属外壳,上部开有两个散热孔。整体形状有圆的也有扁的,许多遥控车的马达是圆的。还有一马达像个竹筒似的,一些太阳帽小风扇里面有。
一般的马达的特点,就是一通电就往一个方向转(正负接反它就反向转),但舵机比普通马达更好玩。360度的舵机与一般马达相似,加电后会往一个方向转动,信号线控制其转动方向和转速;不过我们常说的舵机一般是指 180 度的舵机,这种舵机可以通过信号线让它转动一定的角度,当到达指定的角度后舵机会停下来。
180 度舵机可以:
1、机器人各个关节的运动,可模拟人的关节活动。因为程序可以控制舵机转到指定的角度。
2、遥控小车可以用舵机来拐弯。
3、机械臂 / 机械爪。
4、制作可以转动的摄像头。
5、3D 打印的零部件。
……
只要需要控制转动方向的,都用得上。
舵机依据其工作负荷以及扭力可以有很多种,咱们用来开发做实验的,最好选 9 克舵机。尽管这种舵机力气不大,但是开发板可以直接供电,不用外加电源补电。就是这种,蓝色透明外壳的。
正因为外壳是透明的,所以你能看到里面有什么。当然了,你如果像老周这样喜欢搞破坏的,可以把拆开玩玩。
拆的时候最好戴个手套,毕竟这是电机,你拆过就知道,里面有很多润滑油的。装回去的时候要注意那几个减速齿轮,有的大伙伴一时冲动拆了,然后装不回去。虽然这种舵机一般几块钱一个,但你也不能这么浪费。
舵机里面主要这几个部件:
1、马达。这个肯定有的,不然怎么转。
2、电位器。不同角度会改变不同的电阻值,使控制板能驱动马达转多少圈。
3、控制板。
4、N 个减速齿轮。
上面只是了解一下舵机,接下来看重点——怎么控制角度。一般的教程会跟你说,通过 PWM 设置不同占空比来控制舵机转动角度。这并不算错,因为控制舵机确实是用 PWM 实现的。不过,这样说其实不准确。实际上,让舵机转动多少度是通过高电平的持续时间来控制的。
通常情况下,控制脉冲的周期时长是 20 毫秒,即 20000 微秒。故用PWM时,频率设定 50 Hz(1 / 0.02秒 = 50 Hz)。舵机识别高电平的时间范围为 0.5 ms 到 2.5 ms,即 500 us 到 2500 us。下图是老周盗来的动图。
这就是为什么PWM能控制舵机的原因,在 0.5ms 之前的时间舵机不关心,所以这段时间无论输出信号是高电平还是低电平都可以,因此关注的核心变为高电平什么时候关闭(变为低电平)。
如上图,如果高电平持续到 0.5 毫秒时变为低电平,则舵机旋转至 0 度。
如果高电平在 1.5 毫秒时关闭,则舵机旋转到 90 度。
如果高电平在 2.5 毫秒时关闭,舵机旋转至 180 度。
综上所述,将控制舵机的信号换算为PWM的占空比,需要准备以下条件:
1、周期时间长度,一般为 20000 us,换算为PWM频率为 50Hz。
2、高电平有效的持续时间,一般值为:min = 500us,max = 2500us。
3、500 – 2500 us,其时间段为 2500-500 = 2000us,用180度平分这2000微秒,即每一度角对应的时间为 2000 / 180 ≈ 11.11 us / deg。
综合一下,占空比可这样算:
假设要旋转 90 度,即占空比:
把公式翻译成中文,就是这样
有了上面的基础,写代码就好办了。这里老周写了一个测试程序,这个命令行程序可以通过输入命令来修改参数,这样方便大家做实验。命令帮助信息如下:
using static System.Console; static void ShowHelps() { WriteLine("{0,-15}{1}", "h", "显示帮助信息"); WriteLine("{0,-15}{1}", "x", "退出"); WriteLine("{0,-15}{1}", "d n", "设置周期(微秒)"); WriteLine("{0,-15}{1}", "t n", "高电平起始时间(微秒)"); WriteLine("{0,-15}{1}", "p n", "高电平结束时间(微秒)"); WriteLine("{0,-15}{1}", "a n", "角度"); WriteLine("{0,-15}{1}", "o", "发送脉冲信号"); WriteLine("{0,-15}{1}", "s", "停止脉冲"); WriteLine(); }
下面的代码是计算 1 度角对应的时间长度(微秒)。
static void ComputeMicroToAngle() { microForAngle = (maxMicrosec - minMicrosec) / 180.0d; }
minMicrosec 是高电平持续的最小时间:500us;maxMicrosec 是高电平持续的最大时间:2500us。
这几个变量的声明如下:
/// <summary> /// 高电平持续的最小时间,一般为 500us /// </summary> static int minMicrosec = 500; /// <summary> /// 高电平持续的最大时间,一般为 2500us /// </summary> static int maxMicrosec = 2500; /// <summary> /// 一度角对应的持续时间,单位为微秒(us) /// </summary> static double microForAngle = 0d;
主体代码如下,其余的可以下载源代码查看。
// 创建PWM通道实例 PwmChannel ch = PwmChannel.Create(0, 0); ShowHelps(); //运行后打印一次帮助信息 bool working = true; //用来跳出循环的标志变量 while(working) { Write(">>>"); // 读取键盘输入的一行文本 string line = ReadLine(); // 读取第一个字符 char first = line[0]; // 分析命令 switch(first) { case \'h\': //显示帮助信息 ShowHelps(); break; case \'x\': //跳出循环,退出程序 working = false; break; case \'d\': //设置周期,一般是20000us ch.Frequency = ParseFreq(line[1..].Trim()); break; case \'t\': //设置高电平持续的最小时间 ParseMinMicrosecond(line[1..].Trim()); ComputeMicroToAngle(); break; case \'p\': //设置高电平持续的最大时间 ParseMaxMicrosecond(line[1..].Trim()); ComputeMicroToAngle(); break; case \'a\': //设置要旋转的角度 double angle = ParseAngle(line[1..].Trim()); ch.DutyCycle = ComputeDuty(angle, ch.Frequency); break; case \'o\': //开始发送脉冲 ch.Start(); break; case \'s\': //停止发送脉冲 ch.Stop(); break; default: WriteLine("<<< 无效命令"); break; } } ch.Dispose();
编译,发布,上传到树莓派。注意舵机有三条线:
1、红线(一般在中间),接树莓派 5V 引脚(供电正极)。
2、黑线或棕色线,接树莓派任意一个 GND 引脚(供电负极)。
3、黄色(有的是白色)是信号线,用来控制舵机,接树莓派的 GPIO 18,这个是树莓派全系列通用的默认 PWM 引脚。
运行程序,第一步,输入 d 20000,设定周期(为了统一,所有时间参数都是以微秒为单位)。
输入 t 500 设置高电平控制角度的起始时间,就是持续最小值,取500。
输入 p 2500,设置高电平的结束时间,即持续的最大值,一般取 2500。
输入小写字母 o,开始PWM。
输入 a 120,旋转至 120 度。
输入 a 30 ,旋转至 30 度。
输入小写字母 s ,停止PWM信号,输入 x 退出。
看看效果。
相关源代码,请点这里下载