先前做了一个外包app关于google的blockly二次开发,文见Android 基于Android blockly和蓝牙通信的机器人编程APP,于是乎对于硬件方面也产生了一些兴趣,自己实现了软件,但是硬件还不太懂,于是乎开始学习研究一下硬件方面的知识,也打算自己做一个智能小车。
如果你也是个软件开发者,也希望自己多少能懂一点硬件知识,软硬兼修的话,可以跟着我一起也来制作一个智能小车。期间会遇到很多硬件相关的术语或者专有名词,优秀的你,肯定会不懂就查不懂就学的,这是最重要的过程。
1、蓝牙控制小车的行进 2、小车的速度可调 3、超声波避障 4、循迹
我这里花了点时间把之前的采购清单整理了一下,我当时找了很多家店,那家的比较便宜,分享给大家,不是打广告,只是为了让大家省点钱。(当时前后购买了好几次,运费给了不少,这里大家可以一次性买清)
1、红外循迹传感器 4个 [https://item.taobao.com/item.htm?spm=a1z09.2.0.0.5b4e2e8dE025fh&id=43751274135&_u=aldqfnla542] 2、BLE蓝牙hc-08(带引脚)[https://item.taobao.com/item.htm?spm=a1z09.2.0.0.5b4e2e8dE025fh&id=36426439097&_u=aldqfnl3d00] 3、arduino uno [https://item.taobao.com/item.htm?spm=a1z09.2.0.0.5b4e2e8dE025fh&id=525462013769&_u=aldqfnl5f3a] 4、智能小车底盘及其轮子 [https://item.taobao.com/item.htm?spm=a1z09.2.0.0.5b4e2e8dE025fh&id=43735291557&_u=aldqfnl41dc] 5、L298n电机驱动模块[https://item.taobao.com/item.htm?spm=a1z09.2.0.0.5b4e2e8dE025fh&id=521709080904&_u=aldqfnl6aaa] 6、超声波测距模块[https://item.taobao.com/item.htm?spm=a1z09.2.0.0.5b4e2e8dE025fh&id=555002408091&_u=aldqfnlb0d6] 7、杜邦线 10cm 20cm 30cm 和 母对母 母对公 公对公 每样一排40根 [https://item.taobao.com/item.htm?spm=a1z09.2.0.0.5b4e2e8dE025fh&id=546979596255&_u=aldqfnla016] 8、18950电池盒[https://item.taobao.com/item.htm?spm=a1z09.2.0.0.5b4e2e8dE025fh&id=521746780501&_u=aldqfnl8b94] 9、铜柱 各类长短 带头的 不带头的 都来个10根吧[https://item.taobao.com/item.htm?spm=a1z09.2.0.0.5b4e2e8dE025fh&id=558741728111&_u=aldqfnl0dc7] 10、螺丝 螺帽 若干 多数是3mm的 其他的也可以买点 15根+吧[https://item.taobao.com/item.htm?spm=a1z09.2.0.0.5b4e2e8dE025fh&id=521771839187&_u=aldqfnl2beb] 11、螺丝刀[https://item.taobao.com/item.htm?spm=a1z09.2.0.0.5b4e2e8dE025fh&id=521703689593&_u=aldqfnlead1] 12、mini面包板[https://item.taobao.com/item.htm?spm=a1z09.2.0.0.5b4e2e8dE025fh&id=521761502825&_u=aldqfnl6f02] 13、黑色电工胶带[https://item.taobao.com/item.htm?spm=a1z09.2.0.0.5b4e2e8dE025fh&id=521702965521&_u=aldqfnle13c] 14、sg90舵机[https://item.taobao.com/item.htm?spm=a1z09.2.0.0.5b4e2e8dE025fh&id=43792233123&_u=aldqfnl3bf5] 15、hr-sr04超声波模块支架[https://item.taobao.com/item.htm?spm=a1z09.2.0.0.5b4e2e8dE025fh&id=521750109177&_u=aldqfnlf46b] 16、小电钻(可选)[https://item.taobao.com/item.htm?spm=a1z09.2.0.0.5b4e2e8dE025fh&id=521708346105&_u=aldqfnlac59] 17、充电电池18650 3.7v 2只(我这里是神火,较贵,可买其他的 注意:3.7v的 普通超市是没有的,不是一般的7号或者5号电池 )[https://item.jd.com/5171888.html] 18、充电器[https://item.jd.com/4284307.html]
成型图:(多图预警!!!)
1、首先按照小车的说明书把小车的马达和轮子都依次装上去,如图 2、然后把买来的杜邦线,可以直接拔掉接头,露出线 3、然后接线,左边两排的马达会有连接,右边两排的马达会有连接。注意,这里我们先按照如图所示的方法接线。 4、第一层底盘底部装上3个循迹用的红外传感器,依次排列,9根线 装上之后,第一层底盘的东西就装的差不多了 如图
5、第二层底盘上部,装上电池盒
6、第二层底盘上部,装上舵机。这里使用到了,舵机支架和舵机,有点不太好装,耐心一点
7、把两层底盘合并起来,OK,大功告成!!!!鼓掌👏 最终如图
ps: 这样一次性装好,可以避免多次的拆装底盘(我就是拆了好多次,好费神)
小车装好了,激动了一会儿,但是还不能动,现在就要让它能跑起来 跑起来的话,我们需要使用:电机驱动、电池、arduino的板子
先简单介绍一下电机驱动:
电机1接口 out1 out2 和 电机2接口out3 out4用于和马达的交互,原理就是,给马达的高低电平的不同,使得马达往前转动还是往后转动供电正极:可接受5-12v的电压,和电池盒接在一起,两个3.7v的电池7.4伏,够用地线:和电池盒的地线、arduino的地线连在一起5v:连接arduino的电源线ENA、ENB:调速用的,默认情况下,有个盖子,盖起来的,相当于直接连接,满压5v就代表全速。后面会接arduino的pwm接口,来调速 -In1~4: 接在arduino上接入如图:
1、把马达对应的 4个线接入到电机的out1 out2 out3 out4口, 注意对照一下前面接线的颜色,顺序暂时可是弄成跟我的一样(因为后面arduino的代码里面要一致)
2、连接电池盒与电机(一般我们认为 红色是电源、黑色为地线),连接arduino与电机
3、连接电机的in1-4口和arduino的19、18、17、16(有人问是哪里,A0~A5其实分别对应的是14-19)
连接好,装上电池,接下来解释,写arduino的程序了
1、去官网下载编译器
2、通过电源接口连接到电脑 连接成功之后就有自己板子的信息了,如果没有的话,去网上查一下,下载对应操作系统的arduino的驱动就好
3、代码 这里我们就简单的让小车 循环前后左右动。
#include <Servo.h> //定义五中运动状态 #define STOP 0 #define FORWARD 1 #define BACKWARD 2 #define TURNLEFT 3 #define TURNRIGHT 4 //定义需要用到的引脚 int leftMotor1 = 16; int leftMotor2 = 17; int rightMotor1 = 18; int rightMotor2 = 19; void setup() { // put your setup code here, to run once: //设置控制电机的引脚为输出状态 pinMode(leftMotor1, OUTPUT); pinMode(leftMotor2, OUTPUT); pinMode(rightMotor1, OUTPUT); pinMode(rightMotor2, OUTPUT); } void loop() { // put your main code here, to run repeatedly: int cmd; for(cmd=0;cmd<5;cmd++)//依次执行向前、向后、向左、想有、停止四个运动状态 { motorRun(cmd); delay(2000);//每个命令执行2s } } //运动控制函数 void motorRun(int cmd) { switch(cmd){ case FORWARD: digitalWrite(leftMotor1, LOW); digitalWrite(leftMotor2, HIGH); digitalWrite(rightMotor1, LOW); digitalWrite(rightMotor2, HIGH); break; case BACKWARD: digitalWrite(leftMotor1, HIGH); digitalWrite(leftMotor2, LOW); digitalWrite(rightMotor1, HIGH); digitalWrite(rightMotor2, LOW); break; case TURNLEFT: digitalWrite(leftMotor1, HIGH); digitalWrite(leftMotor2, LOW); digitalWrite(rightMotor1, LOW); digitalWrite(rightMotor2, HIGH); break; case TURNRIGHT: digitalWrite(leftMotor1, LOW); digitalWrite(leftMotor2, HIGH); digitalWrite(rightMotor1, HIGH); digitalWrite(rightMotor2, LOW); break; default: digitalWrite(leftMotor1, LOW); digitalWrite(leftMotor2, LOW); digitalWrite(rightMotor1, LOW); digitalWrite(rightMotor2, LOW); } }OK,不出意外的话,你的小车也可以动起来啦!!!!鼓掌👏
代码解析:
case FORWARD: digitalWrite(leftMotor1, LOW); digitalWrite(leftMotor2, HIGH); digitalWrite(rightMotor1, LOW); digitalWrite(rightMotor2, HIGH); break;可以看到 前进的时候,我们把引脚leftMotor1设置为了低电压,leftMotor2设置为了高电压
有人会问为啥 从图上看,为啥两个马达的 高低电压 不同,却是相同的方向。那是因为,这两个马达 是这样放的。
如果是你把后面那个摆成跟前面那个一样,后面的下上就变成了上下,其实上下的高低电压就一致了。所以,前后轮的转动方向就一致了(这里根据我们的接线方法 我们是前进,如果说接线方法不同或者接反了,可以把代码里面的LOW和HIGH进行互换)
转弯: 我们这里的行为方式是跟坦克类似的,因为我们没有转向轮。 如果想要左转,就是左边轮子向后,右边轮子向前。这样的话会是,原地的左转;如果想要,向左转弯,保持前进的话,就应该是,左边保持低速前进,右边保持高速前进;如果想要调整原地转动的速率和转幅,可以调整左轮的速度和右轮的速度差值来实现。后面的蓝牙小车调速会讲到。 右转也是同理。
接线方法:
TX:接Arduino UNO开发板”RX”引脚 RX:接Arduino UNO开发板”TX”引脚 GND:接Arduino UNO开发板”GND”引脚 VCC:接Arduino UNO开发板”5V”或”3.3V”引脚
这里,我们使用到了调速功能,因为前面说到我们如果想要前进转弯,就需要设置左右两排轮子不同的速度来实现。这里需要电机驱动接入ENA ENB的引脚接口到arduino的PWM接口(arduino上面的引脚号码前有~的就是PWM接口,这里我们使用5、6)。
我们这里如果接入了蓝牙的RXD、TXD的时候,如果想要连接arduino和电脑进行代码写入,是不行的。因为,如果蓝牙占用了RXD和TXD,没有办法写入了。拔掉一个,然后就可以了。
代码:
#include <Servo.h> //定义五中运动状态 String STOP = "D105FFFF"; String UP = "D101FFFF"; String DOWN = "D102FFFF"; String LEFT = "D103FFFF"; String RIGHT = "D104FFFF"; String LEFT_UP = "D111FFFF"; String LEFT_DOWN = "D112FFFF"; String RIGHT_UP = "D113FFFF"; String RIGHT_DOWN = "D114FFFF"; //定义需要用到的引脚 int leftMotor1 = 16; int leftMotor2 = 17; int rightMotor1 = 18; int rightMotor2 = 19; int leftPWM = 5; int rightPWM = 6; int inputPin = 7; // 定义超声波信号接收接口 int outputPin = 8; // 定义超声波信号发出接口 String comdata = ""; void setup() { // put your setup code here, to run once: Serial.begin(9600); //设置控制电机的引脚为输出状态 pinMode(leftMotor1, OUTPUT); pinMode(leftMotor2, OUTPUT); pinMode(rightMotor1, OUTPUT); pinMode(rightMotor2, OUTPUT); //超声波控制引脚初始化 pinMode(inputPin, INPUT); pinMode(outputPin, OUTPUT); //调速 pinMode(leftPWM, OUTPUT); pinMode(rightPWM, OUTPUT); } void loop() { while (Serial.available() > 0) { int i = Serial.read(); char c = char(i); comdata += c; //每次读一个char字符,并相加 delay(2); } if (comdata.length() > 0) { Serial.println(comdata); //打印接收到的字符 motorRun(comdata, 250); comdata = ""; } } //运动控制函数 void motorRun(String cmd, int value) { analogWrite(leftPWM, value); //设置PWM输出,即设置速度 analogWrite(rightPWM, value); if (cmd == UP) { Serial.println("UP"); digitalWrite(leftMotor1, LOW); digitalWrite(leftMotor2, HIGH); digitalWrite(rightMotor1, HIGH); digitalWrite(rightMotor2, LOW); } else if (cmd == DOWN) { Serial.println("DOWN"); digitalWrite(leftMotor1, HIGH); digitalWrite(leftMotor2, LOW); digitalWrite(rightMotor1, LOW); digitalWrite(rightMotor2, HIGH); } else if (cmd == LEFT) { Serial.println("left"); analogWrite(leftPWM, value - 90); //设置PWM输出,即设置速度 analogWrite(rightPWM, value - 70); digitalWrite(leftMotor1, LOW); digitalWrite(leftMotor2, HIGH); digitalWrite(rightMotor1, LOW); digitalWrite(rightMotor2, HIGH); } else if (cmd == RIGHT) { Serial.println("right"); analogWrite(leftPWM, value - 70); //设置PWM输出,即设置速度 analogWrite(rightPWM, value - 90); digitalWrite(leftMotor1, HIGH); digitalWrite(leftMotor2, LOW); digitalWrite(rightMotor1, HIGH); digitalWrite(rightMotor2, LOW); } else if (cmd == LEFT_UP) { analogWrite(leftPWM, value - 150); //设置PWM输出,即设置速度 Serial.println("LEFT_UP"); digitalWrite(leftMotor1, LOW); digitalWrite(leftMotor2, HIGH); digitalWrite(rightMotor1, HIGH); digitalWrite(rightMotor2, LOW); } else if (cmd == LEFT_DOWN) { Serial.println("LEFT_DOWN"); analogWrite(leftPWM, value - 150); //设置PWM输出,即设置速度 digitalWrite(leftMotor1, HIGH); digitalWrite(leftMotor2, LOW); digitalWrite(rightMotor1, LOW); digitalWrite(rightMotor2, HIGH); } else if (cmd == RIGHT_UP) { analogWrite(rightPWM, value - 150); //设置PWM输出,即设置速度 Serial.println("RIGHT_UP"); digitalWrite(leftMotor1, LOW); digitalWrite(leftMotor2, HIGH); digitalWrite(rightMotor1, HIGH); digitalWrite(rightMotor2, LOW); } else if (cmd == RIGHT_DOWN) { analogWrite(rightPWM, value - 150); //设置PWM输出,即设置速度 Serial.println("RIGHT_DOWN"); digitalWrite(leftMotor1, HIGH); digitalWrite(leftMotor2, LOW); digitalWrite(rightMotor1, LOW); digitalWrite(rightMotor2, HIGH); } else { Serial.println("stop"); digitalWrite(leftMotor1, LOW); digitalWrite(leftMotor2, LOW); digitalWrite(rightMotor1, LOW); digitalWrite(rightMotor2, LOW); } }代码解析:
这里我们为了可以直接使用我自己的app,所以把指令直接定义为了和我app里面一样的。APP下载地址一共有9个指令,这里的app里面对于小车行进做了两种操作。一种是上下左右停,一种是摇杆(其实就是 上下左右 左上 右上 左下 右下)。如图: analogWrite(leftPWM, value); //设置PWM输出,即设置速度 analogWrite(rightPWM, value);是左右两边轮子的调速代码
默认我们的对于PWM写入的最大速度值为255,按道理区间为1-255,但是,经过测试发现(根据不同电池情况),如果低于一定值的,就带不动马达转动了(我这里大约是100左右,如果电池没电了可能最小值更高)
原地左转
} else if (cmd == LEFT) { Serial.println("left"); analogWrite(leftPWM, value - 90); //设置PWM输出,即设置速度 analogWrite(rightPWM, value - 70); digitalWrite(leftMotor1, LOW); digitalWrite(leftMotor2, HIGH); digitalWrite(rightMotor1, LOW); digitalWrite(rightMotor2, HIGH);这里我的左转,我不想让它原地的转动的转幅太大,所以我调小了两边的转速
左前 } else if (cmd == LEFT_UP) { analogWrite(leftPWM, value - 150); //设置PWM输出,即设置速度 Serial.println("LEFT_UP"); digitalWrite(leftMotor1, LOW); digitalWrite(leftMotor2, HIGH); digitalWrite(rightMotor1, HIGH); digitalWrite(rightMotor2, LOW);这里,你会发现“左前”,其实代码是在“前进”的代码基础上加了一句调速代码,如我们先前所说,我们要保持前进的同时转弯,就需要向前,并且减速左边的轮子,就可以实现了。
ok,写入了代码,接好RX TX线,装上电池,就可以运行了。蓝牙的蓝色的灯也会亮起来。
这时候,可以使用下载好的app来连接蓝牙小车了。连接成功之后,就可以通过控制界面的两种控制方式来控制我们的小车啦!!!不出意外的话,鼓掌👏!!
超神波避障的主要原理: 利用超声波测距传感器测量小车与障碍之间的距离,小于一定值的时候就停下,然后利用舵机来改变超声波传感器的方向继续测距,如果哪边没有障碍,就往哪边转向来避开障碍,然后再前进。
我们这里需要用到的就是舵机和超声波传感器(及其支架)。舵机,之前装底盘的时候就装上去了,如果那时候没有装的话,现在要装的话就比较困难了。
插线:
超声波有4个引脚,vcc、trig、echo、gnd,vcc和gnd插到arduino上,但是,现在来看,arduino上面已经没有足够的 vcc和gnd的引脚口了,所以,我们要利用 mini面包板来 扩展一些。扩展原理: 这样连接的话,一竖排都是相当于是vcc或者gnd线的接口。
trig和echo分别接到arduino的“8、7”号引脚上面(“Trig”引脚控制超声波发出声波,对应int outputPin=8; “Echo”引脚反应接收到返回声波,对应int inputPin=7;)。大致原理就是 发送一个超声波,然后接受一个超声波,然后通过时间来算得距离
舵机一般为三根线:灰色——GND,红色——VCC,橙色——控制信号。因此我们将灰色色线接到GND,红色线接到“+5V”引脚,橙色线接到“9”号引脚
代码:
#include <Servo.h> //定义五中运动状态 String STOP = "D105FFFF"; String UP = "D101FFFF"; String DOWN = "D102FFFF"; String LEFT = "D103FFFF"; String RIGHT = "D104FFFF"; String LEFT_UP = "D111FFFF"; String LEFT_DOWN = "D112FFFF"; String RIGHT_UP = "D113FFFF"; String RIGHT_DOWN = "D114FFFF"; #define BIZHANG_START "D301FFFF" #define BIZHANG_END "D302FFFF" #define PIN_SERVO 9 //舵机信号控制引脚 //定义需要用到的引脚 int leftMotor1 = 16; int leftMotor2 = 17; int rightMotor1 = 18; int rightMotor2 = 19; int leftPWM = 5; int rightPWM = 6; int inputPin = 7; // 定义超声波信号接收接口 int outputPin = 8; // 定义超声波信号发出接口 Servo myServo; //舵机 String lastCmd; String comdata = ""; void setup() { // put your setup code here, to run once: Serial.begin(9600); //舵机引脚初始化 myServo.attach(PIN_SERVO); //设置控制电机的引脚为输出状态 pinMode(leftMotor1, OUTPUT); pinMode(leftMotor2, OUTPUT); pinMode(rightMotor1, OUTPUT); pinMode(rightMotor2, OUTPUT); //超声波控制引脚初始化 pinMode(inputPin, INPUT); pinMode(outputPin, OUTPUT); //调速 pinMode(leftPWM, OUTPUT); pinMode(rightPWM, OUTPUT); } void loop() { while (Serial.available() > 0) { int i = Serial.read(); char c = char(i); comdata += c; //每次读一个char字符,并相加 delay(2); } if (comdata.length() > 0) { Serial.println(comdata); //打印接收到的字符 if (comdata == BIZHANG_START) { avoidance(); } else if (comdata == BIZHANG_END) { motorRun(STOP, 250); } else { motorRun(comdata, 250); } lastCmd = comdata; comdata = ""; } else { if (lastCmd == BIZHANG_START) { Serial.println("现在是 避障"); avoidance(); } } } void avoidance() { int pos; int dis[3];//距离 motorRun(UP, 255); myServo.write(90); dis[1] = getDistance(); //中间 Serial.println("mid dis:" + dis[1]); if (dis[1] < 30) { motorRun(DOWN,255); delay(300); motorRun(STOP, 0); for (pos = 90; pos <= 150; pos += 1) { myServo.write(pos); // tell servo to go to position in variable 'pos' delay(10); // waits 15ms for the servo to reach the position } dis[2] = getDistance(); //左边 Serial.println("left dis:" + dis[2]); for (pos = 150; pos >= 30; pos -= 1) { myServo.write(pos); // tell servo to go to position in variable 'pos' delay(10); // waits 15ms for the servo to reach the position if (pos == 90) dis[1] = getDistance(); //中间 Serial.println("mid dis:" + dis[1]); } dis[0] = getDistance(); //右边 Serial.println("right dis:" + dis[0]); for (pos = 30; pos <= 90; pos += 1) { myServo.write(pos); // tell servo to go to position in variable 'pos' delay(10); // waits 15ms for the servo to reach the position } if (dis[0] < dis[2]) //右边距离障碍的距离比左边近 { //左转 motorRun(LEFT, 250); delay(500); } else //右边距离障碍的距离比左边远 { //右转 motorRun(RIGHT, 250); delay(500); } } } int getDistance() { digitalWrite(outputPin, LOW); // 使发出发出超声波信号接口低电平2μs delayMicroseconds(2); digitalWrite(outputPin, HIGH); // 使发出发出超声波信号接口高电平10μs,这里是至少10μs delayMicroseconds(10); digitalWrite(outputPin, LOW); // 保持发出超声波信号接口低电平 int distance = pulseIn(inputPin, HIGH); // 读出脉冲时间 distance = distance / 58; // 将脉冲时间转化为距离(单位:厘米) Serial.println("distance:" + distance); //输出距离值 if (distance >= 50) { //如果距离小于50厘米返回数据 return 50; }//如果距离小于50厘米 else return distance; } //运动控制函数 void motorRun(String cmd, int value) { analogWrite(leftPWM, value); //设置PWM输出,即设置速度 analogWrite(rightPWM, value); if (cmd == UP) { Serial.println("UP"); digitalWrite(leftMotor1, LOW); digitalWrite(leftMotor2, HIGH); digitalWrite(rightMotor1, HIGH); digitalWrite(rightMotor2, LOW); } else if (cmd == DOWN) { Serial.println("DOWN"); digitalWrite(leftMotor1, HIGH); digitalWrite(leftMotor2, LOW); digitalWrite(rightMotor1, LOW); digitalWrite(rightMotor2, HIGH); } else if (cmd == LEFT) { Serial.println("left"); analogWrite(leftPWM, value - 90); //设置PWM输出,即设置速度 analogWrite(rightPWM, value - 70); digitalWrite(leftMotor1, LOW); digitalWrite(leftMotor2, HIGH); digitalWrite(rightMotor1, LOW); digitalWrite(rightMotor2, HIGH); } else if (cmd == RIGHT) { Serial.println("right"); analogWrite(leftPWM, value - 70); //设置PWM输出,即设置速度 analogWrite(rightPWM, value - 90); digitalWrite(leftMotor1, HIGH); digitalWrite(leftMotor2, LOW); digitalWrite(rightMotor1, HIGH); digitalWrite(rightMotor2, LOW); } else if (cmd == LEFT_UP) { analogWrite(leftPWM, value - 150); //设置PWM输出,即设置速度 Serial.println("LEFT_UP"); digitalWrite(leftMotor1, LOW); digitalWrite(leftMotor2, HIGH); digitalWrite(rightMotor1, HIGH); digitalWrite(rightMotor2, LOW); } else if (cmd == LEFT_DOWN) { Serial.println("LEFT_DOWN"); analogWrite(leftPWM, value - 150); //设置PWM输出,即设置速度 digitalWrite(leftMotor1, HIGH); digitalWrite(leftMotor2, LOW); digitalWrite(rightMotor1, LOW); digitalWrite(rightMotor2, HIGH); } else if (cmd == RIGHT_UP) { analogWrite(rightPWM, value - 150); //设置PWM输出,即设置速度 Serial.println("RIGHT_UP"); digitalWrite(leftMotor1, LOW); digitalWrite(leftMotor2, HIGH); digitalWrite(rightMotor1, HIGH); digitalWrite(rightMotor2, LOW); } else if (cmd == RIGHT_DOWN) { analogWrite(rightPWM, value - 150); //设置PWM输出,即设置速度 Serial.println("RIGHT_DOWN"); digitalWrite(leftMotor1, HIGH); digitalWrite(leftMotor2, LOW); digitalWrite(rightMotor1, LOW); digitalWrite(rightMotor2, HIGH); } else { Serial.println("stop"); digitalWrite(leftMotor1, LOW); digitalWrite(leftMotor2, LOW); digitalWrite(rightMotor1, LOW); digitalWrite(rightMotor2, LOW); } }超声波:
采用Trig引脚触发,给至少10us的高电平脉冲信号模块自动发送8个40kHz的方波,自动检测是否有信号返回有信号返回,通过Echo引脚输出一个高电平脉冲,高电平脉冲持续的时间就是超声波从发射到反射返回的时间。距离=(高电平脉冲时间*340)/2。(声音在空气中传播速度为340m/s)超声波发出引脚“Trig”为高时对外发出超声波,为保证发出10μs声波,因此在发送之前需要将该引脚拉低,并给他一定反应时间。
digitalWrite(outputPin, LOW); // 使发出发出超声波信号接口低电平2μs delayMicroseconds(2);之后发送10μs超声波
digitalWrite(outputPin, HIGH); // 使发出发出超声波信号接口高电平10μs,这里是至少10μs声波发送之后禁止其继续发送,同时开始检测是否反射回来的声波
digitalWrite(outputPin, LOW); // 保持发出超声波信号接口低电平 int distance = pulseIn(inputPin, HIGH); // 读出脉冲时间pulseIn()单位为微秒,声速344m/s,所以距离cm=344100/1000000pulseIn()/2约等于pulseIn()/58.0 distance= distance/58; // 将脉冲时间转化为距离(单位:厘米)
超声波模块工作受物体表面反射程度影响,并且在传播过程中信号强度容易衰减,因此该模块适用的检测距离有限,一般在50cm以内相对正确,而且我们在避障时不需要检测太远的距离,因此超过50cm以上的都按50cm计算
舵机:
dis[1] = getDistance(); //中间 Serial.println("mid dis:" + dis[1]); if (dis[1] < 30) { motorRun(DOWN,255); delay(300); motorRun(STOP, 0); for (pos = 90; pos <= 150; pos += 1) { myServo.write(pos); // tell servo to go to position in variable 'pos' delay(10); // waits 15ms for the servo to reach the position } dis[2] = getDistance(); //左边 Serial.println("left dis:" + dis[2]); for (pos = 150; pos >= 30; pos -= 1) { myServo.write(pos); // tell servo to go to position in variable 'pos' delay(10); // waits 15ms for the servo to reach the position if (pos == 90) dis[1] = getDistance(); //中间 Serial.println("mid dis:" + dis[1]); } dis[0] = getDistance(); //右边 Serial.println("right dis:" + dis[0]); for (pos = 30; pos <= 90; pos += 1) { myServo.write(pos); // tell servo to go to position in variable 'pos' delay(10); // waits 15ms for the servo to reach the position } if (dis[0] < dis[2]) //右边距离障碍的距离比左边近 { //左转 motorRun(LEFT, 250); delay(500); } else //右边距离障碍的距离比左边远 { //右转 motorRun(RIGHT, 250); delay(500); }先获取中间时候的距离,小于30之后停下。向左边转60度左右,然后到达最左边的时候,获取左边的距离,然后转动到中间获取中间的距离,然后转动到右边,获取右边的距离。 比较左右边距离,哪边长小车就往哪边 转动,然后继续向前。
ok,如果线接好了、代码录入了,装上电池就可以使用app的避障,“开始”和“结束”按钮,来运行了。 不出意外,你的小车就有了简单的避障功能了。故障👏!!!
ps:利用超声波避障,总会有些不太精准的情况,比如障碍不是一个较为平整的能够被很好测算距离的情况。总之,我们追求的是过程,过程最重要嘛。
循迹的话由于循迹算法比较粗糙,有时候会有一些小问题,比如冲出黑线区域等。所以这里,简单的介绍一下。
循迹模块的接线,也比较简单,增加了10,11,12号引脚的接入,和蓝牙控制命令 #define XUNJI_START "D201FFFF" #define XUNJI_END "D202FFFF"
这里可以看到3个循迹的模块,一共9根线,其中每个模块的一个input线就接到arduino对应的引脚号上面,其余的就是电源和地线,接到面包板上面(面包板之前已经讲过原理,不懂的自己查一下)
最后循迹的线可以自己定一下,反正重点就是个T字
ok! 大功告成 😬
PS:
有些人买的蓝牙型号不一致,只要是BLE就行。 不过如果有些CharacteristicSerial不一致的话,需要修改一下 修改地址
public static final String UUID_SERVICE = “0000ffe0-0000-1000-8000-00805f9b34fb”; public static final String UUID_INDICATE = “0000000-0000-0000-8000-00805f9b0000”; public static final String UUID_NOTIFY = “0000ffe1-0000-1000-8000-00805f9b34fb”; public static final String UUID_WRITE = “0000ffe1-0000-1000-8000-00805f9b34fb”; public static final String UUID_READ = “3f3e3d3c-3b3a-3938-3736-353433323130”;
里面就有对应的一些特征序列号。如果不知道的话,卖家那里问问,或者手上一些软件可以查看(连上蓝牙,手机上来看特征序列号)
主要就是READ,WRITE,NOTIFY这几个