零基础入门学用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.xdnf.cn/news/1451455.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

Pytorch环境配置的方法

Pytorch虚拟环境配置全流程 以安装pytorch1.9.1为例 1. 创建虚拟环境 安装Anaconda3&#xff0c;打开 PowerShell 创建虚拟环境并进入&#xff1a; conda create -n torch1.9.1 python3.8 conda activate torch1.9.1 conda create -n torch1.9.1 python3.8 conda activate to…

Flowable-决策表设计器

✨✨✨ 最好用的Flowable决策表设计器 ✨✨✨ 最好用的Flowable流程设计器 本文中内容和案例出自贺波老师的书《深入Activiti流程引擎&#xff1a;核心原理与高阶实战》&#xff0c;书中的介绍更全面、详细&#xff0c;推荐给大家。 深入Activiti流程引擎

3dsMax怎样让渲染效果更逼真出色?三套低中高参数设置

渲染是将精心构建的3D模型转化为逼真图像的关键步骤。但要获得令人惊叹的渲染效果&#xff0c;仅仅依赖默认设置是不够的。 实现在追求极致画面效果的同时&#xff0c;兼顾渲染速度和时间还需要进行一些调节设置&#xff0c;如何让渲染效果更加逼真&#xff1f; 一、全局照明与…

【递归、搜索与回溯】综合练习一

综合练习一 1.找出所有子集的异或总和再求和2.全排列 II3.电话号码的字母组合4.括号生成 点赞&#x1f44d;&#x1f44d;收藏&#x1f31f;&#x1f31f;关注&#x1f496;&#x1f496; 你的支持是对我最大的鼓励&#xff0c;我们一起努力吧!&#x1f603;&#x1f603; 1.找…

sku与spu的区别!!!

一个 SPU 可以有多个 SKU。

深度学习(六)——神经网络的基本骨架:nn.Module的使用

一、torch.nn简介 官网地址&#xff1a; torch.nn — PyTorch 2.0 documentation 1. torch.nn中的函数简介 Containers&#xff1a;神经网络的骨架 Convolution Layers&#xff1a;卷积层 Pooling layers&#xff1a;池化层 Padding Layers&#xff1a;Padding Non-linear …

Linux多线程编程中的同步与互斥

文章目录 一、线程同步与互斥1、理解线程同步2、互斥的概念3、小结 二、互斥锁&#xff08;Mutex&#xff09;1、互斥锁的定义和作用2、pthread库中的互斥锁3、互斥锁的实现原理4、示例代码演示互斥锁的基本用法 三、条件变量&#xff08;Condition Variable&#xff09;1、条件…

RK3566调试VI5301

VI5301是南京芯视界推出的一款直接飞行时间&#xff08;dToF&#xff09;传感器&#xff0c;与ST的VL53L0x兼容。 一、开发平台 系统&#xff1a;linux 4.19(buidroot) 二、驱动移植 解压厂家提供的驱动文件&#xff1a;VI5301_Linux_General_M40_V202&#xff0c;目录结构…

QML学习及实战

QML学习及实战&#xff08;更多内容&#xff09; 创建项目 3. 剩下的就是一路下一步即可 添加静态资源——图片 添加之后完成之后的路径 案列 || demo 可以参考的资料&#xff1a;https://github.com/gongjianbo/MyTestCode/blob/master/README.md 1. 文本省略号 Text {wi…

Python中关于电商商品数据的采集【taobao/JD/商品详情数据返回】

在Python中采集电商商品数据&#xff08;如淘宝、京东等&#xff09;通常涉及到网络爬虫&#xff08;web scraping&#xff09;或称为网络数据抓取&#xff08;web data scraping&#xff09;。由于电商平台通常会有反爬虫机制&#xff0c;因此直接抓取数据可能会遇到各种挑战&…

CV每日论文--2024.6.14

1、ICE-G: Image Conditional Editing of 3D Gaussian Splats 中文标题&#xff1a;ICE-G&#xff1a;3D 高斯斑点的图像条件编辑 简介&#xff1a;近年来,出现了许多技术来创建高质量的3D资产和场景。然而,当涉及到这些3D对象的编辑时,现有方法要么速度慢、要么牺牲质量,要么…

【ai】blender4.1 安装插件

开源软件,所以资料充足插件及配置 下载插件插件是python开发的 编辑中的偏好设置 点击选中 点击一键切换中文英文 切换主题 插件源码

搭建 Redis 集群【Windows】

Redis 集群是一个分布式存储解决方案&#xff0c;它将数据分布在多个Redis节点上&#xff0c;以提高系统的可伸缩性、可靠性和性能。 1. 集群概念与特点 集群概念&#xff1a;Redis集群是由多个相互独立的 Redis 节点组成&#xff0c;这些节点通过高速网络互联&#xff0c;并作…

数据采集项目2-业务数据同步

全量同步 每天都将业务数据库中的全部数据同步一份到数据仓库 全量同步采用DataX datax datax使用 执行 python /opt/module/datax/bin/datax.py /opt/module/datax/job/job.json 更多job.json配置文件在&#xff1a; 生成的DataX配置文件 java -jar datax-config-genera…

Go Module详解

文章目录 基本介绍相关环境变量Go Module的使用初始化项目&#xff08;go mod init&#xff09;管理依赖项&#xff08;go mod edit&#xff09;获取依赖项&#xff08;go mod download&#xff09;整理依赖项&#xff08;go mod tidy&#xff09;导入vendor目录&#xff08;go…

优思学院|做质量没有前途?10年质量人想对大家说...

你是否也有过这样的困惑&#xff1f;做质量工作究竟有没有前途&#xff1f;是不是感觉每天都在重复一样的事情&#xff0c;看不到未来的希望&#xff1f; 今天&#xff0c;优思学院分享一个任职于五百强企业、有着10年经验的质量人、六西格玛黑带学生徐某的文章&#xff0c;和…

【VS】尚未配置为Web项目XXXX指定的本地IIS URL HTTP://localhost

报错原因&#xff1a; 我们在Web项目的属性配置中勾选了“使用本地IIS Web服务器”&#xff1b; 本来嘛&#xff0c;这也没啥&#xff0c;问题是当我们的电脑IP改变时&#xff0c;将会导致程序找不到原来的IP地址了&#xff0c;那么当然会报错啦。 解决办法&#xff1a; 其实…

新一代大核卷积反超ViT和ConvNet!同参数量下性能、精度、速度完胜

大核卷积网络是CNN的一种变体&#xff0c;也是深度学习领域的一种重要技术&#xff0c;它使用较大的卷积核来处理图像数据&#xff0c;以提高模型对视觉信息的理解和处理能力。 这种类型的网络能够捕捉到更多的空间信息&#xff0c;因为它的大步长和大感受野可以一次性覆盖图像…

填报志愿选大学专业,文科生如何选专业?

读文科的同学接触的专业知识相对广泛&#xff0c;往往被认为是“万金油”&#xff0c;他们仿佛什么都能做&#xff0c;但是和专业技能类知识不同&#xff0c;缺乏技术支持&#xff0c;从而使得文科专业的就业方向和前景远远比不上理科专业那么明朗&#xff0c;对于众多文科生而…

加速“芯”动力 | 2024集成电路测试工程师研修班(苏州场)报名通知

6月19日—20日&#xff0c;加速“芯”动力——2024集成电路测试工程师研修班正式开课。本次培训课程内容包括芯片设计测试技术分享、解决方案分享、ATE编程接口介绍、ATE向量微指令、ATE量产界面介绍、测试开发基础培训、程序开发实训等内容&#xff0c;感兴趣的小伙伴&#xf…