舵机实验

罗大富 BigRich大约 5 分钟ESP32Python

舵机在电子产品中非常常见,比如四足机器人、固定翼航模等都有应用,因此学习舵机对后续完成电子制作非常有意义。本节课学习使用 MicroPython 的 PWM 对 SG90 舵机旋转角度控制。

实验原理

舵机是一种位置(角度)伺服的驱动器,适用于那些需要角度不断变化并可以保持的控制系统。舵机只是一种通俗的叫法,其本质是一个伺服电机。

舵机有很多规格,但所有的舵机都有外接三根线,分别用棕、红、橙三种颜色进行区分,由于舵机品牌不同,颜色也会有所差异,棕色为接地线,红色为电源正极线,橙色为信号线。只要通过信号线给予规定的控制信号即可实现舵机码盘的转动。

SG90 的主要电气参数:

  • 使用电压: 4.8V - 6V
  • 尺寸: 221.5mm x 11.8mm x 22.7mm
  • 重量: 9g
  • 角度范围:0-180°

舵机的工作原理是由接收机或者单片机发出信号给舵机,其内部有一个基准电路,将获得的直流偏置电压与电位器的电压比较,获得电压差输出。经由电路板上的 IC 判断转动方向,再驱动无核心马达开始转动,透过减速齿轮将动力传至摆臂,同时由位置检测器送回信号,判断是否已经到达定位。当电机转速一定时,通过级联减速齿轮带动电位器旋转,使得电压差为0,电机停止转动。一般舵机旋转的角度范围是 0 度到 180 度,当然也有 0 度到 360 度。

我们没有必要了解舵机的内部结构,只需要知道如何通过 PWM 控制其转动即可。舵机的控制就是通过一个固定的频率,给其不同的占空比的,来控制舵机不同的转角。

舵机的转动的角度是通过调节 PWM(脉冲宽度调制)信号的占空比来实现的,标准 PWM(脉冲宽度调制)信号的周期固定为 20ms(50Hz),理论上脉宽分布应在 1ms 到 2ms 之间,但是,事实上脉宽可由 0.5ms 到 2.5ms 之间,脉宽和舵机的转角 0°~180° 相对应。有一点值得注意的地方,由于舵机牌子不同,对于同一信号,不同牌子的舵机旋转的角度也会有所不同。

0.5-2.5ms 的 PWM 高电平部分对应控制 180 度舵机的 0-180 度,因此,对应的控制关系是这样的:

高电平占整个周期(20ms)的时间舵机旋转的角度对应的占空比
0.5ms0.5 // 20
1ms45°1 // 20
1.5ms90°1.5 // 20
2ms135°2 // 20
2.5ms180°2.5 // 20

硬件电路设计

物料清单(BOM 表):

材料名称数量
舵机1
杜邦线(跳线)3

注意

注意接线顺序

软件程序设计

上节课我们已经学习了 MicroPython 的 PWM 构造函数和方法,这节课依然可以 PWM。根据实验原理,我们可以直接操作 PWM 值来控制舵机的转动角度,代码如下:

'''
该程序作用是使用 PWM 模块控制舵机转动
在线文档:https://docs.geeksman.com/
'''

import time
from machine import Pin, PWM

# 定义舵机控制对象
my_servo = PWM(Pin(13))
# 定义舵机频率
my_servo.freq(50)

# 使用不同方法控制转动角度

# 使用 duty() 方法转动到 0°,duty 方法的范围是 0-1023,
# 因此,参数值为 1023 // 20 * 0.5 取整等于 25
my_servo.duty(25)
time.sleep(2)

# 使用 duty_u16() 方法转动到 90°,duty_u16 方法的范围是 0-65535,
# 因此,参数值为 65535 // 20 * 1.5 取整等于 4915
my_servo.duty_u16(int(65535//20*2.5))
time.sleep(2)

了解了如何驱动舵机模块了还不够,我们也可以像封装数码管驱动代码一样,自己写一个舵机驱动代码,像舵机这种常用的模块。

MicroPython 拥有着庞大的用户群,自然舵机模块也有开源的代码,直接拿过来使用即可,这就是使用 MicroPython 开发的高效之处,市面上常见的模块在网上几乎都可以找到相应的模块代码,大家一定要善于在网上搜索资源。

我们可以把从网上下载来的代码放到 MicroPython 设备中的 libs 目录下(libs 存放第三方库,common 存放自己写的常用变量,函数,类等等),代码如下:

from machine import PWM
import math

# originally by Radomir Dopieralski http://sheep.art.pl
# from https://bitbucket.org/thesheep/micropython-servo

class Servo:
    """
    A simple class for controlling hobby servos.
    Args:
        pin (machine.Pin): The pin where servo is connected. Must support PWM.
        freq (int): The frequency of the signal, in hertz.
        min_us (int): The minimum signal length supported by the servo.
        max_us (int): The maximum signal length supported by the servo.
        angle (int): The angle between the minimum and maximum positions.
    """
    def __init__(self, pin, freq=50, min_us=600, max_us=2400, angle=180):
        self.min_us = min_us
        self.max_us = max_us
        self.us = 0
        self.freq = freq
        self.angle = angle
        self.pwm = PWM(pin, freq=freq, duty=0)

    def write_us(self, us):
        """Set the signal to be ``us`` microseconds long. Zero disables it."""
        if us == 0:
            self.pwm.duty(0)
            return
        us = min(self.max_us, max(self.min_us, us))
        duty = us * 1024 * self.freq // 1000000
        self.pwm.duty(duty)

    def write_angle(self, degrees=None, radians=None):
        """Move to the specified angle in ``degrees`` or ``radians``."""
        if degrees is None:
            degrees = math.degrees(radians)
        degrees = degrees % 360
        total_range = self.max_us - self.min_us
        us = self.min_us + total_range * degrees // self.angle
        self.write_us(us)

接着在主程序中调用 servo.py 模块中的内容:

'''
该程序作用是使用 PWM 模块控制舵机转动
在线文档:https://docs.geeksman.com/
'''
import time
from machine import Pin
from libs.servo import Servo


# 定义舵机控制对象
my_servo = Servo(Pin(13), max_us=2500)


# 程序入口
if __name__ == '__main__':
    while True:
        my_servo.write_angle(0)
        time.sleep(0.5)
        my_servo.write_angle(45)
        time.sleep(0.5)
        my_servo.write_angle(90)
        time.sleep(0.5)
        my_servo.write_angle(135)
        time.sleep(0.5)
        my_servo.write_angle(180)
        time.sleep(0.5)
上次编辑于:
贡献者: Luo