零基础入门学用Arduino 第二部分(二)

重要的内容写在前面:

  1. 该系列是以up主太极创客的零基础入门学用Arduino教程为基础制作的学习笔记。
  2. 个人把这个教程学完之后,整体感觉是很好的,如果有条件的可以先学习一些相关课程,学起来会更加轻松,相关课程有数字电路(强烈推荐先学数电,不然可能会有一些地方理解起来很困难)、模拟电路等,然后就是C++(注意C++是必学的)
  3. 文章中的代码都是跟着老师边学边敲的,不过比起老师的版本我还把注释写得详细了些,并且个人认为重要的地方都有详细的分析。
  4. 一些函数的介绍有参考太极创客官网给出的中文翻译,为了便于现查现用,把个人认为重要的部分粘贴了过来并做了一些修改。
  5. 如有错漏欢迎指正。

视频链接:2-1 MeArm项目概述_哔哩哔哩_bilibili

太极创客官网:太极创客 – Arduino, ESP8266物联网的应用、开发和学习资料

四、开发机械臂程序

1、准备工作

(1)电路部分按下图所示连接即可。

(2)连接完成后将下面的初始化调整程序下载到开发板中,让舵机转轴转到规定的初始位置。

#include <Servo.h> Servo base, rArm, fArm, claw ;  //建立4个舵机对象void setup() 
{ base.attach(11);     // base 伺服舵机连接引脚11 舵机代号'b'rArm.attach(10);     // rArm 伺服舵机连接引脚10 舵机代号'r'fArm.attach(9);      // fArm 伺服舵机连接引脚9  舵机代号'f'claw.attach(6);      // claw 伺服舵机连接引脚6  舵机代号'c'Serial.begin(9600);
} 
void loop() 
{ base.write(90); // 将base(底盘)舵机设置为初始位置delay(100);rArm.write(90); // 将rArm(后臂)舵机设置为初始位置delay(100);fArm.write(90); // 将fArm(前臂)舵机设置为初始位置delay(100);claw.write(90); // 将claw(钳子)舵机设置为初始位置delay(3000); 
} 

(3)将4个MeArm舵机摇臂按以下示意图装配到舵机上。在MeArm机械臂安装过程中不要让调整好的舵机摇臂转动,如不小心转动了已经调整好的舵机摇臂,需要将摇臂恢复图示状态或使用MeArm舵机初始化调整程序再次对舵机进行初始化调整。

(4)根据图纸或说明书或视频教程安装机械臂,安装完毕后查看电路连接是否出现问题(比如正负极短接、舵机引线接错、舵机未与Arduino共地等问题),然后再运行调试程序,看看机械臂会不会产生异动或者异响,及时调整舵机摇臂的位置或者更换有问题的舵机。

2、通过串口控制机械臂(一步到位)

(1)连接完成后将下面的程序下载到开发板中。

①全局变量及包含的头文件:

#include <Servo.h>                //使用servo库
Servo base, fArm, rArm, claw ;    //创建4个servo对象
//建立4个int型变量存储当前电机角度值,初始角度值为设备启动后初始状态所需要的电机角度数值
int basePos = 90;
int rArmPos = 90;
int fArmPos = 90;
int clawPos = 90;
//存储电机极限值(const指定该数值为常量,常量数值在程序运行中不能改变)
const int baseMIN = 0;
const int baseMAX = 180;
const int rArmMIN = 45;
const int rArmMAX = 180;
const int fArmMIN = 35;
const int fArmMAX = 120;
const int clawMIN = 25;
const int clawMAX = 100;

②初始化工作部分:

void setup()
{base.attach(11);     // base 伺服舵机连接引脚11 舵机代号'b'delay(200);          // 稳定性等待rArm.attach(10);     // rArm 伺服舵机连接引脚10 舵机代号'r'delay(200);          // 稳定性等待fArm.attach(9);      // fArm 伺服舵机连接引脚9  舵机代号'f'delay(200);          // 稳定性等待claw.attach(6);      // claw 伺服舵机连接引脚6  舵机代号'c'delay(200);          // 稳定性等待Serial.begin(9600); Serial.println("Welcome to Taichi-Maker Robot Arm Tutorial.");   
}

③循环工作部分:

void loop()
{//使用串口监视器输入电机指令控制机械臂电机if (Serial.available() > 0) { //指令举例: b45,将底盘舵机调整到45度位置char serialCmd = Serial.read();  //获取串口接收缓存中的一个字符armDataCmd(serialCmd);           //更改所记录的“当前舵机角度”}//根据记录的当前舵机角度进行设置base.write(basePos);delay(10);fArm.write(fArmPos); delay(10);rArm.write(rArmPos); delay(10);claw.write(clawPos);  delay(10);   
}

④更改所记录的“当前舵机角度”:

void armDataCmd(char serialCmd)
{Serial.print("serialCmd = ");Serial.print(serialCmd);  int servoData = Serial.parseInt();   //获取串口接收缓存中的整数数据作为角度值switch(serialCmd)    //根据命令的第一个字符判断需要控制哪个舵机{case 'b':  if(servoData > baseMAX) servoData = baseMAX;  //判断是否越上界if(servoData < baseMIN) servoData = baseMIN;  //判断是否越下界basePos = servoData;  //更改当前舵机角度Serial.print("  Set base servo value: ");Serial.println(servoData);break;case 'c':  if(servoData > clawMAX) servoData = clawMAX;  //判断是否越上界if(servoData < clawMIN) servoData = clawMIN;  //判断是否越下界clawPos = servoData;  //更改当前舵机角度Serial.print("  Set claw servo value: ");Serial.println(servoData);break;case 'f':  if(servoData > fArmMAX) servoData = fArmMAX;  //判断是否越上界if(servoData < fArmMIN) servoData = fArmMIN;  //判断是否越下界fArmPos = servoData;  //更改当前舵机角度Serial.print("  Set fArm servo value: ");Serial.println(servoData);break;case 'r':  if(servoData > rArmMAX) servoData = rArmMAX;  //判断是否越上界if(servoData < rArmMIN) servoData = rArmMIN;  //判断是否越下界rArmPos = servoData;  //更改当前舵机角度Serial.print("  Set rArm servo value: ");Serial.println(servoData);break;case 'o':  reportStatus();break;default:Serial.println(" Unknown Command.");}  
}
void reportStatus()
{Serial.println("");Serial.println("");Serial.println("++++++ Robot-Arm Status Report +++++");Serial.print("Claw Position: clawPos = "); Serial.println(claw.read());Serial.print("Base Position: basePos = "); Serial.println(base.read());Serial.print("Rear  Arm Position: rArmPos = "); Serial.println(rArm.read());Serial.print("Front Arm Position: fArmPos = "); Serial.println(fArm.read());Serial.println("++++++++++++++++++++++++++++++++++++");Serial.println("");
}

(2)然后进行人工调试。

①通过串口助手向Arduino发送内容“b45”,机械臂的base舵机摇臂将立刻旋转至45°的位置,同理可调试其它3个舵机。

②通过串口助手向Arduino发送内容“b200”,由于200°超出base舵机的上界180°,机械臂的base舵机摇臂将立刻旋转至上界180°的位置,同理可调试其它3个舵机的上下界。(需要注意的是,上下界指的是机械臂舵机能达到的不损坏机械臂时的最大/小角度,这个角度可以对四个舵机分别进行调试而得出,每个机械臂的舵机旋转上下界可能略有差异,但只要每个舵机都经过正确的初始化调整,差异应该是很小的)

3、通过串口控制机械臂(有缓慢转动的过程)

(1)在上例中,通过Arduino直接控制舵机旋转,会发现舵机摇臂旋转的速度非常快,然而现实中大多自动工作的机械臂都是缓慢转动的,如果每一个动作都是“一气呵成”,这将增加非常多不必要的麻烦与危险,为了让机械臂缓慢转动,可以将一次大幅度的转动分成若干次小幅度的转动完成,每次小幅度转动间隔一定的时间,这样即可实现机械臂的缓慢转动

(2)连接完成后将下面的初始化调整程序下载到开发板中,然后进行人工调试。

①全局变量及包含的头文件:

#include <Servo.h>                //使用servo库
Servo base, fArm, rArm, claw ;    //创建4个servo对象
//建立4个int型变量存储当前电机角度值,初始角度值为设备启动后初始状态所需要的电机角度数值
int basePos = 90;
int rArmPos = 90;
int fArmPos = 90;
int clawPos = 90;
//存储电机极限值(const指定该数值为常量,常量数值在程序运行中不能改变)
const int baseMIN = 0;
const int baseMAX = 180;
const int rArmMIN = 45;
const int rArmMAX = 180;
const int fArmMIN = 35;
const int fArmMAX = 120;
const int clawMIN = 25;
const int clawMAX = 100;

②初始化工作部分:

void setup()
{base.attach(11);     //base 伺服舵机连接引脚11 舵机代号'b'delay(200);          //稳定性等待rArm.attach(10);     //rArm 伺服舵机连接引脚10 舵机代号'r'delay(200);          //稳定性等待fArm.attach(9);      //fArm 伺服舵机连接引脚9  舵机代号'f'delay(200);          //稳定性等待claw.attach(6);      //claw 伺服舵机连接引脚6  舵机代号'c'delay(200);          //稳定性等待Serial.begin(9600); Serial.println("Welcome to Taichi-Maker Robot Arm Tutorial.");   
}

③循环工作部分:

void loop()
{//使用串口监视器输入电机指令控制机械臂电机if (Serial.available() > 0) { //指令举例: b45,将底盘舵机调整到45度位置char serialCmd = Serial.read();  //获取串口接收缓存中的一个字符armDataCmd(serialCmd);           //更改所记录的“当前舵机角度”}//根据记录的当前舵机角度进行设置base.write(basePos);delay(10);fArm.write(fArmPos); delay(10);rArm.write(rArmPos); delay(10);claw.write(clawPos);  delay(10);   
}

④更改所记录的“当前舵机角度”:(reportStatus函数的实现沿用上例即可)

void armDataCmd(char serialCmd)
{Serial.print("serialCmd = ");Serial.print(serialCmd);  int servoData = Serial.parseInt();  //获取串口接收缓存中的整数数据作为角度值int fromPos, toPos;switch(serialCmd)   //根据命令的第一个字符判断需要控制哪个舵机{case 'b':  fromPos = base.read();    //读取base舵机的当前角度值toPos = servoData;        //命令中的角度值作为调整后角度值if(servoData > baseMAX) servoData = baseMAX;  //判断是否越上界if(servoData < baseMIN) servoData = baseMIN;  //判断是否越下界if (fromPos <= toPos) //如果“起始角度值”小于“目标角度值” ,每15ms向目标转动1°for (int i=fromPos; i<=toPos; i++){base.write(i);delay(15);}else                //否则“起始角度值”大于“目标角度值”,每15ms向目标转动1°for (int i=fromPos; i>=toPos; i--){base.write(i);delay(15);}basePos = servoData;Serial.print("  Set base servo value: ");Serial.println(servoData);break;case 'c':fromPos = claw.read();    //读取claw舵机的当前角度值toPos = servoData;        //命令中的角度值作为调整后角度值if(servoData > clawMAX) servoData = clawMAX;  //判断是否越上界if(servoData < clawMIN) servoData = clawMIN;  //判断是否越下界if (fromPos <= toPos) //如果“起始角度值”小于“目标角度值” ,每15ms向目标转动1°for (int i=fromPos; i<=toPos; i++){claw.write(i);delay(15);}else                //否则“起始角度值”大于“目标角度值”,每15ms向目标转动1°for (int i=fromPos; i>=toPos; i--){claw.write(i);delay(15);}clawPos = servoData;Serial.print("  Set claw servo value: ");Serial.println(servoData);break;  case 'f':  fromPos = fArm.read();    //读取fArm舵机的当前角度值toPos = servoData;        //命令中的角度值作为调整后角度值if(servoData > fArmMAX) servoData = fArmMAX;  //判断是否越上界if(servoData < fArmMIN) servoData = fArmMIN;  //判断是否越下界if (fromPos <= toPos) //如果“起始角度值”小于“目标角度值” ,每15ms向目标转动1°for (int i=fromPos; i<=toPos; i++){fArm.write(i);delay(15);}else                //否则“起始角度值”大于“目标角度值”,每15ms向目标转动1°for (int i=fromPos; i>=toPos; i--){fArm.write(i);delay(15);}fArmPos = servoData;Serial.print("  Set fArm servo value: ");Serial.println(servoData);break;case 'r':  fromPos = rArm.read();    //读取rArm舵机的当前角度值toPos = servoData;        //命令中的角度值作为调整后角度值if(servoData > rArmMAX) servoData = rArmMAX;  //判断是否越上界if(servoData < rArmMIN) servoData = rArmMIN;  //判断是否越下界if (fromPos <= toPos) //如果“起始角度值”小于“目标角度值” ,每15ms向目标转动1°for (int i=fromPos; i<=toPos; i++){rArm.write(i);delay(15);}else                //否则“起始角度值”大于“目标角度值”,每15ms向目标转动1°for (int i=fromPos; i>=toPos; i--){rArm.write(i);delay(15);}rArmPos = servoData;Serial.print("  Set rArm servo value: ");Serial.println(servoData);break;case 'o': reportStatus();break;default: Serial.println(" Unknown Command.");}  
}

(3)然后进行人工调试。

①通过串口助手向Arduino发送内容“b45”,机械臂的base舵机摇臂将缓慢地旋转至45°的位置,同理可调试其它3个舵机。

②通过串口助手向Arduino发送内容“b200”,由于200°超出base舵机的上界180°,机械臂的base舵机摇臂将缓慢地旋转至180°的位置,同理可调试其它3个舵机的上下界。(需要注意的是,每个机械臂的舵机旋转上下界可能略有差异,但只要每个舵机都经过正确的初始化调整,差异应该是很小的)

4、通过串口控制机械臂(有设置快捷指令)

(1)电路连接完成后将下面的程序下载到开发板中。

①全局变量及包含的头文件:

#include <Servo.h>                //使用servo库
Servo base, fArm, rArm, claw ;    //创建4个servo对象//存储电机极限值(const指定该数值为常量,常量数值在程序运行中不能改变)
const int baseMin = 0;
const int baseMax = 180;
const int rArmMin = 45;
const int rArmMax = 180;
const int fArmMin = 35;
const int fArmMax = 120;
const int clawMin = 25;
const int clawMax = 100;int DSD = 15; //Default Servo Delay (默认电机运动延迟时间)
//此变量用于控制电机运行速度,增大此变量数值将降低电机运行速度,从而控制机械臂动作速度

②初始化工作部分:

void setup()
{base.attach(11);     //base 伺服舵机连接引脚11 舵机代号'b'delay(200);          //稳定性等待rArm.attach(10);     //rArm 伺服舵机连接引脚10 舵机代号'r'delay(200);          //稳定性等待fArm.attach(9);      //fArm 伺服舵机连接引脚9  舵机代号'f'delay(200);          //稳定性等待claw.attach(6);      //claw 伺服舵机连接引脚6  舵机代号'c'delay(200);          //稳定性等待base.write(90); delay(10);     //base 伺服舵机旋转角度初始化+稳定性等待fArm.write(90); delay(10);     //fArm 伺服舵机旋转角度初始化+稳定性等待rArm.write(90); delay(10);     //rArm 伺服舵机旋转角度初始化+稳定性等待claw.write(90); delay(10);     //claw 伺服舵机旋转角度初始化+稳定性等待Serial.begin(9600); Serial.println("Welcome to Taichi-Maker Robot Arm Tutorial");   
}

③循环工作部分:

void loop()
{if (Serial.available() > 0) {  char serialCmd = Serial.read();  //获取指令中的第一个字符armDataCmd(serialCmd);           //根据串行指令执行相应操作}
}void armDataCmd(char serialCmd)
{if (serialCmd == 'b' || serialCmd == 'c' || serialCmd == 'f' || serialCmd == 'r')  //如果第一个字符是舵机代号{int servoData = Serial.parseInt();    //获取指令中的整数数据servoCmd(serialCmd, servoData, DSD);  //调用机械臂舵机运行函数(参数:舵机名,目标角度,单次延迟时间)} else {switch(serialCmd){    case 'o':  //输出舵机状态信息reportStatus();break;case 'i':  //机械臂初始化armIniPos();break;     default:   //未知指令反馈Serial.println("Unknown Command.");}}  
}

④机械臂舵机运行函数:

void servoCmd(char servoName, int toPos, int servoDelay)
{  Servo servo2go;  //创建servo对象//串口监视器输出接收指令信息Serial.println("");Serial.print("+Command: Servo ");Serial.print(servoName);Serial.print(" to ");Serial.print(toPos);Serial.print(" at servoDelay value ");Serial.print(servoDelay);Serial.println(".");Serial.println("");  int fromPos; //建立变量,存储电机起始运动角度值switch(servoName)   //根据命令的第一个字符判断需要控制哪个舵机{case 'b':if(toPos >= baseMin && toPos <= baseMax){  //判断是否越界,越界就报错servo2go = base;        //把对象base拷贝到servo2gofromPos = base.read();  //获取当前base电机角度值用于“电机运动起始角度值”break;} else {Serial.println("+Warning: Base Servo Value Out Of Limit!");return;}case 'c':if(toPos >= clawMin && toPos <= clawMax){  //判断是否越界,越界就报错servo2go = claw;        //把对象claw拷贝到servo2gofromPos = claw.read();  //获取当前claw电机角度值用于“电机运动起始角度值”break;} else {Serial.println("+Warning: Claw Servo Value Out Of Limit!");return;}case 'f':if(toPos >= fArmMin && toPos <= fArmMax){  //判断是否越界,越界就报错servo2go = fArm;        //把对象fArm拷贝到servo2gofromPos = fArm.read();  //获取当前fArm电机角度值用于“电机运动起始角度值”break;} else {Serial.println("+Warning: fArm Servo Value Out Of Limit!");return;}case 'r':if(toPos >= rArmMin && toPos <= rArmMax){  //判断是否越界,越界就报错servo2go = rArm;        //把对象rArm拷贝到servo2gofromPos = rArm.read();  //获取当前rArm电机角度值用于“电机运动起始角度值”break;} else {Serial.println("+Warning: rArm Servo Value Out Of Limit!");return;}  }//通过对象servo2go指挥电机运行if (fromPos <= toPos) //如果“起始角度值”小于“目标角度值”for (int i=fromPos; i<=toPos; i++){servo2go.write(i);delay(servoDelay);}else                  //否则“起始角度值”大于“目标角度值”for (int i=fromPos; i>=toPos; i--){servo2go.write(i);delay(servoDelay);}
}

⑤报告舵机当前角度函数:

void reportStatus()
{Serial.println("");Serial.println("");Serial.println("+ Robot-Arm Status Report +");Serial.print("Claw Position: "); Serial.println(claw.read());Serial.print("Base Position: "); Serial.println(base.read());Serial.print("Rear  Arm Position:"); Serial.println(rArm.read());Serial.print("Front Arm Position:"); Serial.println(fArm.read());Serial.println("++++++++++++++++++++++++++");Serial.println("");
}

⑥机械臂重新初始化函数:

void armIniPos()
{Serial.println("+Command: Restore Initial Position.");int robotIniPosArray[4][3] =   //使用二维数组存储4个舵机的初始化信息{/*  舵机代号 目标角度 单次延迟  */{    'b',     90,    DSD},{    'r',     90,    DSD},{    'f',     90,    DSD},{    'c',     90,    DSD} };   for (int i = 0; i < 4; i++)  //调用4次机械臂舵机运行函数,分别初始化4个舵机{servoCmd(robotIniPosArray[i][0], robotIniPosArray[i][1], robotIniPosArray[i][2]);}
}

(2)然后进行人工调试。

①通过串口助手向Arduino发送内容“b45”,机械臂的base舵机摇臂将缓慢地旋转至45°的位置,同理可调试其它3个舵机。

②通过串口助手向Arduino发送内容“b200”,由于200°超出base舵机的上界180°,机械臂的base舵机不会有任何动作,同时Arduino通过串口报指令有误,同理可调试其它3个舵机。

③通过串口助手向Arduino发送内容“o”,Arduino将通过串口发送四个舵机当前的状态。

④通过串口助手向Arduino发送内容“i”,Arduino将控制四个舵机恢复初始状态。

5、通过串口控制机械臂(设有手柄控制方式)

(1)电路连接完成后将下面的程序下载到开发板中。

①全局变量及包含的头文件:

#include <Servo.h>                //使用servo库
Servo base, fArm, rArm, claw ;    //创建4个servo对象//存储电机极限值(const指定该数值为常量,常量数值在程序运行中不能改变)
const int baseMin = 0;
const int baseMax = 180;
const int rArmMin = 45;
const int rArmMax = 180;
const int fArmMin = 35;
const int fArmMax = 120;
const int clawMin = 25;
const int clawMax = 100;int DSD = 15; //Default Servo Delay (默认电机运动延迟时间)
//此变量用于控制电机运行速度,增大此变量数值将降低电机运行速度,从而控制机械臂动作速度bool mode;          //记录当前的模式:mode = 1 —— 指令模式,mode = 0 —— 手柄模式
int moveStep = 3;  //每一次按下手柄按键的舵机移动量(仅适用于手柄模式)

②初始化工作部分:

void setup()
{base.attach(11);     //base 伺服舵机连接引脚11 舵机代号'b'delay(200);          //稳定性等待rArm.attach(10);     //rArm 伺服舵机连接引脚10 舵机代号'r'delay(200);          //稳定性等待fArm.attach(9);      //fArm 伺服舵机连接引脚9  舵机代号'f'delay(200);          //稳定性等待claw.attach(6);      //claw 伺服舵机连接引脚6  舵机代号'c'delay(200);          //稳定性等待base.write(90); delay(10);     //base 伺服舵机旋转角度初始化+稳定性等待fArm.write(90); delay(10);     //fArm 伺服舵机旋转角度初始化+稳定性等待rArm.write(90); delay(10);     //rArm 伺服舵机旋转角度初始化+稳定性等待claw.write(90); delay(10);     //claw 伺服舵机旋转角度初始化+稳定性等待Serial.begin(9600); Serial.println("Welcome to Taichi-Maker Robot Arm Tutorial");   
}

③循环工作部分:

void loop()
{if (Serial.available()>0) {  char serialCmd = Serial.read();  //获取指令中的第一个字符if(mode == 1)  //根据mode判断现在处于什么模式{armDataCmd(serialCmd);  //指令模式} else {armJoyCmd(serialCmd);   //手柄模式}}
}

④指令模式下的处理逻辑:

void armDataCmd(char serialCmd)
{//判断用户是否因搞错模式而输入错误的指令信息(即指令模式下输入手柄按键信息)if (   serialCmd == 'w' || serialCmd == 's' || serialCmd == 'a' || serialCmd == 'd'|| serialCmd == '5' || serialCmd == '4' || serialCmd == '6' || serialCmd == '8' ){Serial.println("+Warning: Robot in Instruction Mode..."); delay(100);while(Serial.available() > 0) char wrongCommand = Serial.read();  //清除串口缓存中的错误指令return;}                if (serialCmd == 'b' || serialCmd == 'c' || serialCmd == 'f' || serialCmd == 'r'){int servoData = Serial.parseInt();servoCmd(serialCmd, servoData, DSD);  //调用机械臂舵机运行函数(参数:舵机名,目标角度,单次延迟)} elseswitch(serialCmd){   case 'm':    //切换至手柄模式 mode = 0; Serial.println("Command: Switch to Joy-Stick Mode.");break;case 'o':    //输出舵机状态信息reportStatus();break;case 'i':    //机械臂初始化armIniPos();break;  default:     //未知指令反馈Serial.println("Unknown Command.");}  
}

⑤手柄模式下的处理逻辑:

void armJoyCmd(char serialCmd)
{//判断用户是否因搞错模式而输入错误的指令信息(即手柄模式下输入舵机指令)if (serialCmd == 'b' || serialCmd == 'c' || serialCmd == 'f' || serialCmd == 'r'){Serial.println("+Warning: Robot in Joy-Stick Mode...");delay(100);while(Serial.available()>0) char wrongCommand = Serial.read();  //清除串口缓存中的错误指令return;} int baseJoyPos, rArmJoyPos, fArmJoyPos, clawJoyPos;switch(serialCmd){case 'a':  //Base向左Serial.println("Received Command: Base Turn Left");                baseJoyPos = base.read() - moveStep;  //目标角度=当前角度-单次操作移动角度servoCmd('b', baseJoyPos, DSD);break; //调用机械臂舵机运行函数 case 'd':  //Base向右Serial.println("Received Command: Base Turn Right");                baseJoyPos = base.read() + moveStep;  //目标角度=当前角度+单次操作移动角度servoCmd('b', baseJoyPos, DSD);break; //调用机械臂舵机运行函数        case 's':  //rArm向下Serial.println("Received Command: Rear Arm Down");                rArmJoyPos = rArm.read() + moveStep;  //目标角度=当前角度+单次操作移动角度servoCmd('r', rArmJoyPos, DSD);break; //调用机械臂舵机运行函数              case 'w':  //rArm向上Serial.println("Received Command: Rear Arm Up");     rArmJoyPos = rArm.read() - moveStep;  //目标角度=当前角度-单次操作移动角度servoCmd('r', rArmJoyPos, DSD);break; //调用机械臂舵机运行函数  case '8':  //fArm向上Serial.println("Received Command: Front Arm Up");        fArmJoyPos = fArm.read() + moveStep;  //目标角度=当前角度+单次操作移动角度servoCmd('f', fArmJoyPos, DSD);break; //调用机械臂舵机运行函数  case '5':  //fArm向下Serial.println("Received Command: Front Arm Down");        fArmJoyPos = fArm.read() - moveStep;  //目标角度=当前角度-单次操作移动角度servoCmd('f', fArmJoyPos, DSD);break; //调用机械臂舵机运行函数  case '4':  //Claw关闭Serial.println("Received Command: Claw Close Down");        clawJoyPos = claw.read() + moveStep;  //目标角度=当前角度+单次操作移动角度servoCmd('c', clawJoyPos, DSD);break; //调用机械臂舵机运行函数  case '6':  //Claw打开Serial.println("Received Command: Claw Open Up");     clawJoyPos = claw.read() - moveStep;  //目标角度=当前角度-单次操作移动角度servoCmd('c', clawJoyPos, DSD);break; //调用机械臂舵机运行函数  case 'm':   //切换至指令模式 mode = 1; Serial.println("Command: Switch to Instruction Mode.");break;case 'o':   //输出舵机状态信息reportStatus();break;case 'i':   //机械臂初始化armIniPos();break;default:    //未知指令反馈Serial.println("Unknown Command.");return;}  
}

⑥报告舵机当前角度函数:

void reportStatus()
{Serial.println("");Serial.println("");Serial.println("+ Robot-Arm Status Report +");Serial.print("Claw Position: "); Serial.println(claw.read());Serial.print("Base Position: "); Serial.println(base.read());Serial.print("Rear  Arm Position:"); Serial.println(rArm.read());Serial.print("Front Arm Position:"); Serial.println(fArm.read());Serial.println("++++++++++++++++++++++++++");Serial.println("");
}

⑦机械臂重新初始化函数:

void armIniPos()
{Serial.println("+Command: Restore Initial Position.");int robotIniPosArray[4][3] =   //使用二维数组存储4个舵机的初始化信息{/*  舵机代号 目标角度 单次延迟  */{    'b',     90,    DSD},{    'r',     90,    DSD},{    'f',     90,    DSD},{    'c',     90,    DSD} };   for (int i = 0; i < 4; i++)  //调用4次机械臂舵机运行函数,分别初始化4个舵机{servoCmd(robotIniPosArray[i][0], robotIniPosArray[i][1], robotIniPosArray[i][2]);}
}

⑧机械臂舵机运行函数:

void servoCmd(char servoName, int toPos, int servoDelay)
{  Servo servo2go;  //创建servo对象//串口监视器输出接收指令信息Serial.println("");Serial.print("+Command: Servo ");Serial.print(servoName);Serial.print(" to ");Serial.print(toPos);Serial.print(" at servoDelay value ");Serial.print(servoDelay);Serial.println(".");Serial.println("");  int fromPos; //建立变量,存储电机起始运动角度值switch(servoName){case 'b':if(toPos >= baseMin && toPos <= baseMax){  //判断是否越界,越界就报错servo2go = base;        //把对象base拷贝到servo2gofromPos = base.read();  //获取当前base电机角度值用于“电机运动起始角度值”break;} else {Serial.println("+Warning: Base Servo Value Out Of Limit!");return;}case 'c':if(toPos >= clawMin && toPos <= clawMax){  //判断是否越界,越界就报错servo2go = claw;        //把对象claw拷贝到servo2gofromPos = claw.read();  //获取当前claw电机角度值用于“电机运动起始角度值”break;} else {Serial.println("+Warning: Claw Servo Value Out Of Limit!");return;}case 'f':if(toPos >= fArmMin && toPos <= fArmMax){  //判断是否越界,越界就报错servo2go = fArm;        //把对象fArm拷贝到servo2gofromPos = fArm.read();  //获取当前fArm电机角度值用于“电机运动起始角度值”break;} else {Serial.println("+Warning: fArm Servo Value Out Of Limit!");return;}case 'r':if(toPos >= rArmMin && toPos <= rArmMax){  //判断是否越界,越界就报错servo2go = rArm;        //把对象rArm拷贝到servo2gofromPos = rArm.read();  //获取当前rArm电机角度值用于“电机运动起始角度值”break;} else {Serial.println("+Warning: rArm Servo Value Out Of Limit!");return;}  }//通过对象servo2go指挥电机运行if (fromPos <= toPos) //如果“起始角度值”小于“目标角度值”for (int i=fromPos; i<=toPos; i++){servo2go.write(i);delay(servoDelay);}else                  //否则“起始角度值”大于“目标角度值”for (int i=fromPos; i>=toPos; i--){servo2go.write(i);delay(servoDelay);}
}

(2)根据程序注释进行人工调试。

6、配合HC-06蓝牙模块控制机械臂

(1)按照下图所示将电路连接好。

(2)沿用上例的程序即可,手机连接上HC-06蓝牙模块,接着打开配套软件的手柄操作界面,设置好每个键所对应的指令信息,在机械臂处于手柄操作模式的前提下对其进行调试。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/852667.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

通用大模型与垂直大模型:双轨并进的人工智能未来

在人工智能(AI)的浩瀚宇宙中&#xff0c;大模型以其强大的学习能力和广泛的适用性&#xff0c;正逐步成为推动技术进步和产业革新的核心动力。在这股浪潮中&#xff0c;通用大模型与垂直大模型如同两颗璀璨的星辰&#xff0c;各自散发着独特的光芒&#xff0c;共同照亮了AI发展…

VirtualBox配置双网卡实现宿主机和虚拟机相互访问以及虚拟机外网访问

目录 一&#xff1a;背景 二&#xff1a;实现 三&#xff1a;总结 一&#xff1a;背景 在VirtualBox中配置虚拟机以实现本地主机远程登录、访问外网以及虚拟机之间的相互访问&#xff0c;是一种常见的虚拟化实践&#xff0c;适用于多种场景&#xff0c;如开发、测试和远程工…

openh264 帧内预测编码原理:WelsMdI4x4Fast 函数

介绍 说明&#xff1a;该函数内部原理类似WelsMdI4x4函数&#xff0c;具体可以参考&#xff1a;openh264 帧内预测编码原理&#xff1a;WelsMdI4x4 函数。 功能&#xff1a;针对4x4像素块的帧内模式决策的快速实现逻辑原型&#xff1a; int32_t WelsMdI4x4Fast (sWelsEncCtx…

react的自定义组件

// 自定义组件(首字母必须大写) function Button() {return <button>click me</button>; } const Button1()>{return <button>click me1</button>; }// 使用组件 function App() {return (<div className"App">{/* // 自闭和引用自…

React Redux

React Redux是Redux的官方React UI绑定层。它允许您的React组件从Redux存储读取数据&#xff0c;并将操作分派到存储以更新状态。redux是一个管理状态数据state的容器。提供了可预测的状态管理。 React Redux 8.x需要React 16.8.3或更高版本/Rect Native 0.59或更高&#xff0c…

在AMD GPU上加速大型语言模型的Flash Attention

Accelerating Large Language Models with Flash Attention on AMD GPUs — ROCm Blogs 引言 在这篇博客文章中&#xff0c;我们将指导您如何在AMD GPU上安装Flash Attention&#xff0c;并提供与在PyTorch中标准SDPA比较其性能的基准测试。我们还将测量Hugging Face中多个大型…

【Java】解决Java报错:FileNotFoundException

文章目录 引言1. 错误详解2. 常见的出错场景2.1 文件路径错误2.2 文件名拼写错误2.3 文件权限问题2.4 文件路径未正确拼接 3. 解决方案3.1 检查文件路径3.2 使用相对路径和类路径3.3 检查文件权限3.4 使用文件选择器 4. 预防措施4.1 使用配置文件4.2 使用日志记录4.3 使用单元测…

上网行为管理的作用是什么?有哪些上网行为管理软件?

上网行为管理在现代企业及家庭环境中扮演着至关重要的角色&#xff0c;其作用不仅限于提升网络安全性&#xff0c;还涉及保护企业信息安全、提高员工工作效率等多个方面。以下将详细阐述上网行为管理的作用&#xff0c;并介绍几款主流的上网行为管理软件。 一、上网行为管理的作…

Neo4j 桌面版打不开踩坑贴

真的踩坑。。。没有人告诉我为啥桌面版和社区版不能一起下啊&#xff01;&#xff01; 我是先下载了社区版之后再下载的桌面版&#xff0c;结果桌面版界面一直打不开。 尝试了网上多种办法都没效果&#xff0c;好多都是说jdk不兼容导致无法打开&#xff0c;让我从JDK 17 ->…

AUTOSAR平台中的信息安全标准模块

面向MCU端的AUTOSAR CP平台加密组件——Crypto ECU中所有的软件单元都遭受到信息安全攻击的可能。AUTOSAR为保障ECU信息和数据安全&#xff0c;定义了CRYPTO 组件,包含 SecOC、KeyM、IdsM、Csm、CryIf 和Crypto Driver 等标准模块。CRYPTO组件提供各种加解密算法以及密钥管理功…

物联网对智慧驾考应用的价值

随着物联网技术的快速发展&#xff0c;传统行业正经历着前所未有的变革。在智慧驾考领域&#xff0c;4G DTU&#xff08;数据传输单元&#xff09;和工业路由器的应用&#xff0c;不仅提升了考试的规范性和效率&#xff0c;更为驾考行业带来了深远影响。作为工业物联网的资深工…

JVM 类加载器的工作原理

JVM 类加载器的工作原理 类加载器&#xff08;ClassLoader&#xff09;是一个用于加载类文件的子系统&#xff0c;负责将字节码文件&#xff08;.class 文件&#xff09;加载到 JVM 中。Java 类加载器允许 Java 应用程序在运行时动态地加载、链接和初始化类。 2. 类加载器的工…

今年的就业环境不容乐观,你想好怎么应对了吗

今年的就业环境不容乐观&#xff0c;你想好怎么应对了吗 毕业生进入职场的历程往往充满挑战和未知&#xff0c;尤其是在当前经济环境下&#xff0c;失业问题愈发凸显。本文通过分享几位年轻人的真实经历&#xff0c;剖析大学生及职场人士面临的困境&#xff0c;并提供应对策略…

手把手带你搭建一个语音对话机器人,5分钟定制个人AI小助手(新手入门篇)

写在前面 如果你的身边有一个随时待命、聪明绝顶的AI小助手&#xff0c;能够听懂你的话&#xff0c;理解你的需求&#xff0c;用温暖的声音回应你&#xff0c;会是一种什么体验&#xff1f; 今天&#xff0c;带大家一起搭建一个语音对话机器人&#xff0c;拥有一个专属的个人…

games101作业7光线追踪 含多线程和微表面提高

对于光线追踪进行综合运用。 光线与三角形求交 其它的emit那些&#xff0c;现在先不用管&#xff0c;后面看看作用是什么。 inline Intersection Triangle::getIntersection(Ray ray) {Intersection inter;if (dotProduct(ray.direction, normal) > 0)//光线从里面打&…

[Shell编程学习路线]——深入理解Shell编程中的变量(理论与实例)

&#x1f3e1;作者主页&#xff1a;点击&#xff01; &#x1f6e0;️Shell编程专栏&#xff1a;点击&#xff01; ⏰️创作时间&#xff1a;2024年6月12日11点40分 &#x1f004;️文章质量&#xff1a;95分 文章目录 ————前言———— 1 自定义变量 &#x1fae0;…

Zynq学习笔记--AXI4-Stream到视频输出IP是如何工作的?

目录 1. 简介 2. 原理详解 2.1 示例工程 2.2 AXI4-Stream to Video Out 3. Master/Slave Timing Mode 3.1 Slave Timing Mode 3.2 Master Timing Mode 4. 总结 1. 简介 本文主要介绍了 AXI4-Stream 到视频输出 的内容。其中&#xff0c;示例工程展示了一个具体的设计&…

GitLab教程(五):高效的工作模式——Feature Branching

文章目录 1.什么是Feature Branching2.Feature Branching的Git实践 1.什么是Feature Branching 特性分支&#xff08;Feature Branching&#xff09;是一种软件开发工作流&#xff0c;尤其在使用Git或其他版本控制系统时被广泛采用。这种策略鼓励开发者为每一个新功能、改进或…

推荐一款好用的读论文软件操作方法

步骤&#xff1a; 1. 使用一译 —— 文档和论文翻译、对照阅读、讨论和社区 2.上传自己想要翻译的论文即可。 示例 Planing论文双语翻译 1.1 Parting with Misconceptions about Learning-based Vehicle Motion Planning 中英文对照阅读 1.2 Rethinking Imitation-based Pl…

SCT82A32 是一款 100V 电压模式控制同步降压控制器

主要特征 ◦ 5.5V-100V 宽输入范围 ◦ 0.8V-60V 可调输出电压 ◦ 0.8V1% 参考电压 ◦ 最低占空比下的40ns 最小 tON ◦ 最高占空比下的150ns 最小 tOFF • 100 KHz 到 1.2 MHz 开关频率 ◦ 时钟同步输入/输出功能 ◦ 可选择二极管仿真或 FPWM • 7.5V 门极驱动器 ◦ 2.3A …