看过了很多前辈写的文章,但始终感觉前辈们单独看的话教的有些零散,很多知识讲不全而且文章中有些不起眼的小错误,但是把所有前辈的文章综合来看,却又能成体系,所以我打算写这篇文章把前辈们的知识总结概括修正一下,本文尤其适合稍稍有些基础但是对操作的部分感觉比较不熟练,容易忘记的同学复习用。
●PWM是用占空比不同的方波,来模拟“模拟输出”的一种方式。
●舵机的控制一般需要一个20ms左右的时基脉冲,该脉冲的高电平部分一般为0.5ms-2.5ms范围内的角度控制脉冲部分,总间隔为2ms。
●舵机的转动的角度是通过调节PWM(脉冲宽度调制)信号的占空比来实现的。需要使用Arduino上的PWM口控制(数字前带~的),本文使用的是10号引脚。以180度角度伺服为例,那么对应的控制关系是这样的:
脉宽 | 效果(旋转到) |
---|---|
0.5ms | 0度 |
1.0ms | 45度 |
1.5ms | 90度 |
2.0ms | 135度 |
2.5ms | 180度 |
如果是270度舵机
脉宽 | 效果(旋转到) |
---|---|
0.5ms | 0度 |
1.5ms | 135度 |
2.5ms | 270度 |
其他角度与脉宽对应同理。
还有一种连续转动舵机,同样由脉冲信号控制,脉冲宽度0.5ms-2.5ms(500-2500us)同一般舵机,没有固定旋转的角度。
脉宽 | 效果 |
---|---|
500us-1500us | 越小反向速度越快 |
1500us-2500us | 越大正向速度越快 |
说白了我们要通过Arduino的能发送PWM信号的引脚向舵机的信号线(一般是橙色线)发送PWM信号,这个信号携带了控制舵机旋转角度的信息。
那么哪些引脚可以发送PWM信号呢?以NANO为例,从以下两张图片我们可以清晰地看出那些有波浪线引出的引脚是PWM引脚(Pin)。
舵机一般为三条线,红线接5V,除了红线和橙色线以外的线(棕色或黑色)接GND。
#include // 声明调用Servo.h库
#define SERVO_PIN 10//宏定义舵机控制引脚
Servo myservo; //创建一个舵机类,命名为myservo
void setup()
{
myservo.attach(SERVO_PIN);
}
void loop()
{
myservo.write(90);
delay(1000);
}
#include
#define SERVO_PIN 10 //10脚就是你的的舵机信号线一般是橙色线的链接引脚。
Servo myservo;
void setup()
{
myservo.attach(PIN_SERVO);
}
void loop()
{
myservo.write(90);
delay(1000);
}
被标记的内容是你想让舵机转起来之前必须要做的事。
而之后我们可以用write()直接键入角度,比如这里的90,也可以通过writeMicroseconds()键入脉宽,脉宽与角度有关,单位us。
相关函数:
1.servo类成员函数
attach() 设定舵机的接口,只有9或10接口可利用
write() 用于设定舵机旋转角度的语句,可设定的角度范围是0°到180°
writeMicroseconds() 用于设定舵机PWM的语句,直接用微秒作为参数
read() 用于读取舵机角度的语句,可理解为读取最后一条write()命令中的值
attach() 判断舵机参数是否已发送到舵机所在接口
detach() 使舵机与其接口分离,该接口(9或10)可继续被用作PWM接口
程序如下 ,这是源程序的link
#include // 声明调用Servo.h库
Servo myservo; //创建一个舵机类,命名为myservo
#define SERVO_PIN 10 //宏定义舵机控制引脚
unsigned int PWM = 0; //变量pwm用来存储舵机角度位置,PWM的500对应0度,2500对应舵机的最大角度
//(180度舵机2500对应180度,270度舵机2500对应270度)
void singleServoControl(){
for(PWM = 50; PWM <2450; PWM += 5){ //舵机从50状态转到2450,每次增加5
myservo.writeMicroseconds(PWM); //给舵机写入PWM
delay(10); //延时10ms让舵机转到指定位置
}
for(PWM = 2450; PWM>50; PWM-=5){
myservo.writeMicroseconds(PWM);
delay(10);
}
}
void setup(){
//put your setup code here, to run once:
myservo.attach(SERVO_PIN); // 将10引脚与声明的舵机对象连接起来
}
void loop(){
//put your main code here, to run repeatedly:
singleServoControl();
}
void singleServoControl()是原文作者自己写的函数,为了防止C语言基础差的同学看不懂,本文对此稍作简化,可以更直观一些看到writeMicroseconds()起的作用。
#include
#define SERVO_PIN 10
Servo myservo;
void setup()
{
myservo.attach(SERVO_PIN);
}
void loop()
{
myservo.writeMicroseconds(1500);
delay(1000);
}
write()与writeMicroseconds()有各自擅长的应用场景,一般非连续转动的舵机可以write(),连续转动的舵机可以使用writeMicroseconds(),其中非连续转动大于180度的舵机也可以使用writeMicroseconds()直接输入脉宽,也可以使用map函数把角度比如从(0,270),360度舵机同理,映射到(0,180),然后write出这个角度。 map函数可以看下文,也可以直接跳到文末基础知识部分有个链接,直接点击即可。
写了个例程如下:
#include
Servo myservo;
int angle;
int val=90;
void setup(){
myservo.attach(10);
Serial.begin(9600);
}
void loop(){
angle=map(val,0,270,0,180);
myservo.write(angle);
Serial.print(angle,DEC);
delay(10000);
}
把90度从(0,270)映射到(0,180),从串口得到旋转角度为60。关于串口的知识我放在文末有一个链接,是笔者自己写的,欢迎围观~。
最后值得一提的是当你attach了你的PWM引脚后,并不需要将这个引脚设置成OUTPUT。
看到中文社区的一个大佬的帖子,想稍微讲一下。
链接: Arduino驱动舵机,不调用库函数.
int sp1=10;//定义舵机接口数字接口10
int pulsewidth;//定义脉宽变量
int val;
int val1;
int myangle1;
void setup()
{
pinMode(sp1,OUTPUT);//设定舵机接口为输出接口
//设置两组串口波特率
Serial.begin(9600);
delay(500);
Serial.println("servu=o_seral_simple ready" ) ;
}
void loop()//将0到9的数转化为0到180角度,并让LED闪烁相应数的次数
{
val=Serial.read();//读取串行端口的值
if(val>'0'&&val<='9')
{
val1=val-'0';//将特征量转化为数值变量
val1=map(val1,0,9,0,180);//将角度转化为500-2480的脉宽值
Serial.print("moving servo to ");
Serial.print(val1,DEC);
Serial.println();
for(int i=0;i<=50;i++)//给予舵机足够的时间让它转到指定角度
{
servopulse(sp1,val1);//引用脉冲函数
}
}
//下面是servopulse函数部分(此函数意思:也就是說每次都是0.5ms高電平 1.98ms低電平 然後再0.52ms低電平 17ms延時也是低電平)
void servopulse(int sp1,int val1)//定义一个脉冲函数
{
myangle1=map(val1,0,180,500,2480);
digitalWrite(sp1,HIGH);//将舵机接口电平至高
delayMicroseconds(myangle1);//延时脉宽值的微秒数
digitalWrite(sp1,LOW);//将舵机接口电平至低
delay(20-val1/1000);
}
//servopulse函数部分到此结束
}
map函数的本质是赋值,属于某个区间的A变量等比例转化至另一个区间后赋给B,比如某值在A区间的1/2处,用map映射到B区间后就是位于B区间的1/2处的值。详解在文末链接。
val1=val-‘0’ val是你向串口输入的值但这个数值被Aduino 识别为这个数值对应的ASCII码,如你输入9实际从串口打印出来的值是39(‘9’),而‘0’等于30,用‘9’-‘0’我们得到int 9从而把‘9’转化成9,这就是源程序中说的将特征量转化为数值变量,通过这种方式使val1是一个位于[0,9]的数字。关于串口大家可以看我放在文末的串口链接。
后来这个val1被从[0,9]映射到[0,180],用于从串口打印出角度方便查看。再后来val1从[0,180]映射到[500,2480],很明显这组区间代表着PWM的脉宽单位是us,delayMicroseconds(myangle1)控制着PWM信号中高电平时间,val1/1000把val1的单位转化成ms后可以用放在delay()中,用整个周期20ms减去它作为低电平的时间,至此完成了整个PWM信号的内容,配合串口内容我们实现由上位机通过串口控制舵机。
16路舵机驱动板示例 需要的库文件也可根据这个链接的内容安装,本文不再赘述。如果你已经安装了库文件,直接阅读下文即可。
PCA9685
电压:舵机供电5-7v,接受高一点的电压。
大多数的舵机设计电压都是在5~6V,尤其在多个舵机同时运行时,跟需要有大功率的电源供电。如果直接使用Arduino
5V引脚直接为舵机供电,会出现一些难以预测的问题,所以我们建议你能有个合适的外部电源为驱动板供电。逻辑电路电压:3-5V
通信接口:使用I2C通信,就是SCL、SDA引脚
7位的I2C地址为:0x40 +A5:A0,A5到A0如果不做任何处理的话是0,想要把哪一位置1就把那个引脚焊到一起。另外用i2cdetect检测出还有一个0x70地址一直存在,这是一个通用地址,可以给所有从机下达指令。
OE反使能脚:这个引脚低电平使能,不接的话模块内部默认已经接地使能了,所以正常使用可以不接。
工作频率:40-1000HZ
注意:以下部分对原文也就是上边这个链接的内容做出一些修正!
舵机的控制一般需要一个20ms的时基脉冲,该脉冲的高电平部分一般为0.5ms~2.5ms范围内的角度控制脉冲部分。以180度角度舵机为例,那么对应的控制关系是这样的:
0.5ms————–0度;
1.0ms————45度;
1.5ms————90度;
2.0ms———–135度;
2.5ms———–180度;
PCA9685可以设置更新频率,时基脉冲周期20ms相当于50HZ更新频率。PCA9685采用12位的寄存器来控制PWM占比,对于0.5ms,相当于0.5/20*4096=102的寄存器值。以此类推如下:
0.5ms————–0度:0.5/20 * 4096 = 102
1.0ms————45度:1/20 * 4096 = 204
1.5ms————90度:1.5/20 * 4096 = 306
2.0ms———–135度:2/20 * 4096 = 408
2.5ms———–180度:2.5/20*4096 =510
上文中如:0.5/20、 1/20等均为占空比值,可自行百度。
但是实际使用的时候,还是有偏差,除了0度以及180度,其他需要乘以0.915系数。最后的寄存器值如下:
0.5ms————–0度:0.5/20 * 4096 = 102
1.0ms————45度:1/20 * 4096 = 204 * 0.915 = 187
1.5ms————90度:1.5/20 * 4096 = 306 * 0.915 = 280
2.0ms———–135度:2/20 * 4096 = 408 * 0.915 = 373
2.5ms———–180度:2.5/20 * 4096 =510
控制程序使用串口通讯接受指令,实现0/45/90/135/180度,总共5种角度的控制,我们也可以自己根据需要计算其他角度的寄存器值,在程序开头定义出来。
#include
#include
// 默认地址 0x40
Adafruit_PWMServoDriver pwm = Adafruit_PWMServoDriver();
#define SERVO_0 102
#define SERVO_45 187
#define SERVO_90 280
#define SERVO_135 373
#define SERVO_180 510
// our servo # counter
uint8_t servonum = 0;
char comchar;
void setup() {
Serial.begin(9600);
Serial.println("8 channel Servo test!");
pwm.begin();
pwm.setPWMFreq(50); // 50HZ更新频率,相当于20ms的周期
delay(10);
}
void loop() {
while(Serial.available()>0){
comchar = Serial.read();//读串口第一个字节
switch(comchar)
{
case '0':
pwm.setPWM(0, 0, SERVO_0);
Serial.write(comchar);
break;
case '1':
pwm.setPWM(0, 0, SERVO_45);
Serial.write(comchar);
break;
case '2':
pwm.setPWM(0, 0, SERVO_90);
Serial.write(comchar);
break;
case '3':
pwm.setPWM(0, 0, SERVO_135);
Serial.write(comchar);
break;
case '4':
pwm.setPWM(0, 0, SERVO_180);
Serial.write(comchar);
break;
default: //匹配失败
Serial.write(comchar);
break;
}
}
}
#include
#include
// 默认地址 0x40
Adafruit_PWMServoDriver pwm = Adafruit_PWMServoDriver();
#define SERVO_0 102
#define SERVO_45 187
#define SERVO_90 280
#define SERVO_135 373
#define SERVO_180 510
// our servo # counter
uint8_t servonum = 0;
char comchar;
void setup() {
Serial.begin(9600);
Serial.println(“8 channel Servo test!”);
pwm.begin();
pwm.setPWMFreq(50); // 50HZ更新频率,相当于20ms的周期
delay(10);
}
void loop() {
while(Serial.available()>0){
comchar= Serial.read();//读串口第一个字节
switch(comchar)
{
case ‘0’:
pwm.setPWM(0, 0, SERVO_0);
Serial.write(comchar);
break;
}
}
}
标记的部分是需要我们记忆的。
需要向Arduino发送值
本程序用了一个经典的while(Serial.available()>0)和Serial.read()组合,这说明我们必须要想Arduino的串口里输入值,由后边的switch case结构我们可以看出程序的用意是要我们向串口输入0到4,这5个值又分别控制这接入驱动板0号位置的舵机转过的角度。pwm.setPWM(0, 0, SERVO_0)中第一个0是指驱动板上的舵机接口位置,而SERVO_0是前文中提到的寄存器值,本程序中已经被 #define SERVO_0 102赋值为102,我们也可以直接输入数字102或者其他你已经算好的数值,计算方法前文已经提到。至于中间的0,本文不再介绍,想了解的同学可以前往该链接16路PWM舵机驱动板(PCA 9685) + Arduino。
值得一提的还有驱动板供电和舵机供电都在驱动板上,这两个是独立的,也就是你需要为驱动板接入2个电源,不给舵机供电舵机是绝对不会转的,笔者刚拿到这个PCA9685时以为给板供电的同时就也给舵机供电了,实际上不是这样的。本程序接线如该表:
驱动板 | Arduino | 电源 |
---|---|---|
SDA | A4 | |
SCL | A5 | |
VCC | 5V | |
V+ (板上总共有两个,这个V+是螺丝固定的V+,用来给舵机供电) | 5-7V | |
GND | GND | GND |
为什么要选A4、A5呢?如下图,A4、A5分别又代表SDA与SCL,用于和其他组件进行I2C通信。
上面有出现uint8_t uint16_t,那它们是什么?其实很简单:
1字节 uint8_t //无符号整型
2字节 uint16_t
map函数简介
串口知识
Arduino delayMicroseconds()函数
delayMicroseconds()函数接受单个整数(或数字)参数。此数字表示时间,以微秒为单位。一毫秒内有一千微秒,一秒内有一百万微秒。
目前,可以产生精确延迟的最大值是16383。这可能会在未来的Arduino版本中改变。对于超过几千微秒的延迟,应该使用delay()函数。
delayMicroseconds()函数语法
delayMicroseconds (us) ;
其中, us 是要暂停的微秒数(无符号整型)。
也就是一般延迟时间很短的时候我们常常使用delayMicroseconds (us) 。
下边这个框里的内容可以跳过,放在这里只是因为笔者觉得有些值得琢磨的地方。。。想看的话粘贴到别的地方就看不到删除线了。。。
(1)舵机的追随特性
假设现在舵机稳定在A点,这时候CPU发出一个PWM信号,舵机全速由A点转向B点,在这个过程中需要一段时间,舵机才能运动到B点。
保持时间为Tw 当Tw≥△T时,舵机能够到达目标,并有剩余时间; 当Tw≤△T时,舵机不能到达目标;
理论上:当Tw=△T时,系统最连贯,而且舵机运动的最快。 实际过程中w不尽相同,连贯运动时的极限△T比较难以计算出来。
当PWM信号以最小变化量即(1DIV=8us)依次变化时,舵机的分辨率最高,但是速度会减慢。
PWM信号控制精度制定 如果采用的是 8 位单片机,其数据分辨率为256,那么经过舵机极限参数实验,得到应该将其划分为 250 份。 那么
0.5mS—2.5Ms 的宽度为 2mS = 2000uS。2000uS÷250=8uS,则:PWM 的控制精度为 8us。我们可以以 8uS 为单位递增控制舵机转动与定位。 舵机可以转动 185 度,那么185 度÷250=0.74 度,则:舵机的控制精度为 0.74 度。
1 DIV = 8us ; 250DIV=2ms时基寄存器内的数值为:(#01H)01 ----(#0FAH)250。 共 185 度,分为 250 个位置,每个位置叫 1DIV。则:185÷250 = 0.74 度 / DIV PWM 上升沿函数: 0.5ms + N×DIV 0us ≤ N×DIV ≤ 2ms 0.5ms ≤ 0.5ms+N×DIV ≤ 2.5ms
本文参考了许多文章,文中均已指明出处,再次感谢优秀的前辈们。