舵机是一种位置(角度)伺服的驱动器,适用于那些需要角度不断变化并可以保持的控制系统。在高档遥控玩具,如飞机、潜艇模型,遥控机器人中已经得到了普遍应用。
从 https://gitee.com/billyzh/esp32-cpp-lesson 下载本教程的源码到本地硬盘文件夹,如d:\esp32-cpp-lesson
在VSCode中,选择【文件】->【打开文件夹...】选择上一步保存的文件夹打开
打开项目后,选择config.h文件,修改第10行为
#define APP_LESSON52 1
打开unit5-lesson52/board_config文件,设置器件使用的引脚,
#define MANUAL_BUTTON_PIN GPIO_NUM_39
#define SG90_PIN GPIO_NUM_14
配置启用舵机
#define CONFIG_USE_SERVO 1
创建驱动器件实例,代码如下(unit5-lesson52/my_board.cpp):
MyBoard::MyBoard() : Board() {
Log::Info(TAG, "===== Create Board ...... =====");
Log::Info(TAG, "initial led.");
led_ = new GpioLed(BUILTIN_LED_PIN, false);
// 创建按键对象
std::shared_ptrbutton = std::make_shared(kManualButton, MANUAL_BUTTON_PIN, true, false);
button->BindAction(ButtonAction::Click);
button->BindAction(ButtonAction::DoubleClick);
AddButton(button);
// 创建执行器件对象
std::shared_ptrservo_ptr = std::make_shared(SG90_PIN);
AddActuator(kSG90, servo_ptr);
Log::Info( TAG, "===== Board config completed. =====");
}
程序解读
1. 创建一个按钮Button类实例,绑定单击(Click)、双击(DoubleClick)动作;
2. 创建一个舵机驱动器实例,这里使用内置的舵机驱动类ServoDriver;
舵机驱动器类ServoDriver,代码如下(src/framework/peripheral/servo_driver.h):
class ServoDriver : public Actuator {
public:
ServoDriver(gpio_num_t pin, uint16_t max_angle=180)
: servo_pin_(pin), max_angle_(180) {
servo_ = new Servo();
servo_->attach(servo_pin_);
}
virtual ~ServoDriver() {
servo_->detach();
};
void SetAngle(uint16_t angle) {
if (angle > max_angle_) {
angle = max_angle_;
}
servo_->write(angle);
}
uint16_t GetAngle() {
return servo_->read();
}
void Reset() {
servo_->write(max_angle_ / 2); // 归位
}
void Move(uint16_t angle) {
uint16_t new_angle = servo_->read() + angle;
if (new_angle < 0) new_angle = 0; if (new_angle > max_angle_) new_angle = max_angle_;
servo_->write(new_angle);
}
private:
Servo *servo_;
const gpio_num_t servo_pin_;
const uint16_t max_angle_;
};
ServoDriver类继承自Actuator类,封装了三方库ESP32Servo,提供复位(Reset)、绝对角度(SetAngle)、相对角度(Move)等方法。
控制舵机
舵机控制代码如下(unit5-lesson51/my_application.cpp):
void MyApplication::OnInit() {
std::shared_ptr<Actuator> act_ptr = Board::GetInstance().GetActuator(kSG90);
std::shared_ptr<ServoDriver> servo_ptr = std::static_pointer_cast<ServoDriver>(act_ptr);
servo_ptr->Reset();
}
void MyApplication::OnLoop() {
MyBoard *board = static_cast<myboard*>(&Board::GetInstance());
board->ButtonTick();
delay(1);
}
bool MyApplication::OnPhysicalButtonEvent(const std::string& button_name, const ButtonAction action) {
if (button_name == kManualButton) {
if (action == ButtonAction::Click) {
StepMove(5);
return true;
} else if (action == ButtonAction::DoubleClick) {
StepMove(-5);
return true;
}
}
return Application::OnPhysicalButtonEvent(button_name, action);
}
void MyApplication::StepMove(int step) {
std::shared_ptr<Actuator> act_ptr = Board::GetInstance().GetActuator(kSG90);
std::shared_ptr<ServoDriver> servo_ptr = std::static_pointer_cast<ServoDriver>(act_ptr);
Schedule([servo_ptr, step](){
servo_ptr->Move(step);
});
}
程序解读
1. 在OnInit方法中调用舵机驱动类的Reset方法进行复位;
2. 在OnLoop方法中调用MyBoard的ButtonTick事件,以捕捉按键动作;
3. 重载Application类的OnPhysicalButtonEvent方法响应按键动作,本程序设计为单击向右5度,双击向左5度;
4. StepMove方法中调用舵机驱动类的Move方法进行移动,这里调用Schedule方法来调度Move方法,保障多次调用可以顺序执行;
此处使用static_pointer_cast来进行智能指针的向下强制转换(父类指针转为子类指针),若指针与预期类型不符,返回空指针。
编译项目并上传开发板检验
作业:
实现一个自定义的执行器类。
麦克纳姆轮的移动方式是基于一个有许多位于机轮周边的轮轴的中心轮的原理上,这些成角度的周边轮轴把一部分的机轮转向力转化到一个机轮法向力上面。
MimiClaw 是一款基于 ESP32-S3 芯片的超轻量级AI助手,适合嵌入式AI与物联网开发者快速部署本地化AI代理。本系列教程基于MimiClaw的Arduino移植版本进行讲解,小节主要讲解部署和测试。
就像我们用手机打开WiFi功能后可以浏览附近的可用WiFi。要将手机连接到热点,通常需要打开Wi-Fi设置应用程序,列出可用的网络,然后选择所需的热点。然后输入密码(或不输入密码),可以使用ESP32进行相同的操作。
本文本介绍配置飞书机器人为MimiClaw的一个输入/输出端,和添加一个控制WS2812与LED的控制技能。
一块 30 块钱的开发板 + 一个大模型 API,就能做出可以听懂人话的智能硬件。 本文记录完整安装过程和踩坑经验,确保你跟着做就能跑通。
本文将从手绘架构图入手,逐层拆解 MimiClaw 的分层设计、核心模块、数据流转与底层实现,带你解剖这只“智能虾”的技术骨架,看懂在 C 语言加持下,AI 智能体如何以可穿戴设备的形态,在你身边稳稳运行、离线服务、主动响应。
本文介绍如何在不脱离 ArduinoIDE 可视化开发的前提下,通过一个名为 platform.local.txt 的小文件,实现对 ESP32 编译流程的精准控制。
本文将系统分析程序体积增长的五大根源,并提供经过验证的优化方案,帮助减小固件大小。
本文所DIY的语音助手设备端使用的是MicroPython、服务端是Python,对于很多开发者来说MicroPython入门没难度。
本小节使用音频开发框架实现一个音频录制到文件的示例。
I2S协议通过BCLK、LRCLK和DATA三线精准传输音频数据,但时序边沿、帧格式、时钟源等细节常引发噪声或断连。本文详解ESP32的I2S实现,从协议原理到ESP-IDF v5.x代码配置,助你避开常见陷阱,确保音频稳定传输。