智能小车实践教程

前言

先前做了一个外包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、去官网下载编译器
长这样,我是mac的,不过操作方法都差不多

2、通过电源接口连接到电脑
选择这个uno
连接成功之后就有自己板子的信息了,如果没有的话,去网上查一下,下载对应操作系统的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进行互换)

转弯:
我们这里的行为方式是跟坦克类似的,因为我们没有转向轮。
如果想要左转,就是左边轮子向后,右边轮子向前。这样的话会是,原地的左转;如果想要,向左转弯,保持前进的话,就应该是,左边保持低速前进,右边保持高速前进;如果想要调整原地转动的速率和转幅,可以调整左轮的速度和右轮的速度差值来实现。后面的蓝牙小车调速会讲到。
右转也是同理。

蓝牙小车

我们会用到4个引脚

接线方法:

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面包板来 扩展一些。

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);
  }
}
代码解析:

超声波:

  1. 采用Trig引脚触发,给至少10us的高电平脉冲信号
  2. 模块自动发送8个40kHz的方波,自动检测是否有信号返回
  3. 有信号返回,通过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:利用超声波避障,总会有些不太精准的情况,比如障碍不是一个较为平整的能够被很好测算距离的情况。总之,我们追求的是过程,过程最重要嘛。

循迹

循迹的话由于循迹算法比较粗糙,有时候会有一些小问题,比如冲出黑线区域等。所以这里,简单的介绍一下。

上一下所有的代码
#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 XUNJI_START "D201FFFF"
#define XUNJI_END "D202FFFF"

#define PIN_SERVO 9  //舵机信号控制引脚

//定义需要用到的引脚
int leftMotor1 = 16;
int leftMotor2 = 17;
int rightMotor1 = 18;
int rightMotor2 = 19;

int follow1 = 10;
int follow2 = 11;
int follow3 = 12;

int leftPWM = 5;
int rightPWM = 6;

int inputPin = 7; // 定义超声波信号接收接口
int outputPin = 8; // 定义超声波信号发出接口

int xunjiSpeed = 170;

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);
  //寻迹模块引脚初始化
  pinMode(follow1, INPUT);
  pinMode(follow2, INPUT);
  pinMode(follow3, INPUT);
}

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 == XUNJI_START) {
      follow();
    } else if (comdata == XUNJI_END) {
      motorRun(STOP, 250);
    } else if (comdata == BIZHANG_START) {
      avoidance();
    } else if (comdata == BIZHANG_END) {
      motorRun(STOP, 250);
    } else {
      motorRun(comdata, 250);
    }
    lastCmd = comdata;
    comdata = "";
  } else {
    if (lastCmd == XUNJI_START) {
      Serial.println("现在是 循迹");
      follow();
    } else if (lastCmd == BIZHANG_START) {
      Serial.println("现在是 避障");
      avoidance();
    }
  }
}

void follow()
{
  int data[3];
  data[0] = digitalRead(follow1);
  data[1] = digitalRead(follow2);
  data[2] = digitalRead(follow3);

  //data[x] == 0的时候说明检测到了 黑线
  if (data[0] && data[1] && data[2])//3个都检测到黑线说明 走到了T字  停止
  {
    Serial.println("我stop了现在-");
    motorRun(STOP, 0);
  } else if (!data[0] && data[1] && !data[2] ) { //中间监测到黑线 就直线
    Serial.println("我UP了现在-");
    motorRun(UP, 100);
  } else if (data[0] && !data[1] && !data[2] ) {
Serial.println("我LEFT了现在-");    
    motorRun(LEFT, 180);
  } else if (!data[0] && !data[1] && data[2]) {
    Serial.println("我RIGHT了现在-");
    motorRun(RIGHT, 180);
  } else {
    motorRun(UP, 100);
    Serial.println("不知道干啥-");
  }

  Serial.println("00:" + data[0]);
  Serial.println("---");
  Serial.println("11:" + data[1]);
  Serial.println("---");
  Serial.println("22:" + data[2]);
}

void avoidance()
{
  int pos;
  int dis[3];//距离
  motorRun(UP, xunjiSpeed);
  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);
  }
}

循迹模块的接线,也比较简单,增加了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这几个

Jafir
关注 关注
  • 11
    点赞
  • 125
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
校内实训项目智能小车
03-23
本系统能实现对小车运动状态进接控制行驶及其他的方式。本系统能实现 对小车运动状态进实时控制。系统灵活可靠,精度高满足对的各项要求实时控制。 首先通过控制A9板实现对Arduino的串口通信,通过高低电平控制芯片L298N驱动电机的正向和反向的转动。
智能小车实习报告由STC单片机控制
11-26
这是我的实习报告,很有意思,我想的没做成,倒反让老师买了一个圆梦小车的模型跟着练,不过也学着点东西,唉,我的大学生活,它是其于STC12C5410AD的小车,能直线和寻迹,
智能小车程序代码
08-11
本段代码可在keil下编译开发,代码主要实现智能小车的一系列高级动作
项目七:智能小车
番茄的博客
03-18 456
这两节里配置有误,正确的应该是PSC=7199,ARR=199,大家注意修正!对应源代码:smartCar_project11_1。对应源代码:smartCar_project11_2。对应源代码:smartCar_project9_1。对应源代码:smartCar_project9_2。对应源代码:smartCar_project9_3。对应源代码:smartCar_project1。对应源代码:smartCar_project2。对应源代码:smartCar_project11。
哇靠靠,这也行?零基础DIY无人驾驶小车(三)
cjnewstar111的专栏
02-06 3307
原理: 之前在教程(一)中我们讨论了制作无人驾驶小车的方案,结论是当小车采集到图片之后,通过网络传给笔记本,然后笔记本通过深度学习推理,得出结果,反馈给小车。但是目前通过一些优秀的开源库,将cnn网络优化之后在手机或者树莓派这样的终端上面也有比较好的效果。所以这里真正开始实践的时候,我采用了树莓派的方式,摒弃了笔记本,直接通过树莓派进行图片的采集和深度学习推理,并实时的控制小车的运动。这样的话极...
智能车改舵机中值步骤_智能小车实践教程
weixin_39655689的博客
12-19 1176
前言先前做了一个外包app关于google的blockly二次开发,文见Android 基于Android blockly和蓝牙通信的机器人编程APP,于是乎对于硬件方面也产生了一些兴趣,自己实现了软件,但是硬件还不太懂,于是乎开始学习研究一下硬件方面的知识,也打算自己做一个智能小车。如果你也是个软件开发者,也希望自己多少能懂一点硬件知识,软硬兼修的话,可以跟着我一起也来制作一个智能小车。期间会遇...
树莓派智能小车教程.pdf
02-27
《树莓派智能小车教程》是一份详细...总的来说,这份教程旨在帮助初学者从零开始构建一个具备摄像头、避障和追踪功能的智能小车,通过实践学习树莓派的硬件连接、系统安装和编程控制,提升动手能力和物联网应用技能。
arduino智能车组装步骤-Arduino自动避障智能小车制作教程.pdf
02-27
《Arduino自动避障智能小车制作教程》 本教程主要介绍如何使用Arduino UNO R3制作一款具有自动避障功能的智能小车。通过集成L298N电机驱动模块、超声波测距模块、9G舵机以及相应的硬件连接,我们可以构建一个能够...
51智能小车视频教程 主控制板功能
06-14
### 51智能小车视频教程之主控制板功能详解 #### 一、主控板总体功能 51智能小车的主控板是整个小车的核心部件,它集成了多种功能模块,使得用户能够方便地开发出各种智能小车应用。下面是主控板的主要功能: 1. ...
智能小车设计指南升级版:原理解析与实践教程
智能小车设计指导是一本由河海大学计算机与信息学院(常州)学生科协编写的教程,旨在帮助学生理解和实践现代智能小车的设计过程。第二版相较于第一版进行了升级和改进,主要包括以下几个关键要点: 1. 功能扩展:新...
sumorobot-android-bluetooth:使用 Google Blockly 通过蓝牙在您的 Android 设备上对您的 Sumorobot 进行编程
06-23
sumorobot-android-蓝牙 使用 Google Blockly 通过蓝牙在您的 Android 设备上对您的 Sumorobot 进行编程 安装 安装 Android SDK sudo apt-get install nodejs sudo npm install -g cordova cordova plugin add org.apache.cordova.globalization cordova plugin add org.apache.cordova.dialogs cordova platform update android cordova run android 学分^_^
智能小车新手详细入门教程
11-17
送给 所有爱好电子的 童鞋们 个人觉得是一套不错的新手入门的 教程 里面有很多 基础级资料 非常非常好!蔽障 循迹 等等。。。。加上源程序 设计图 各种完美!
vjc编程无人驾驶小车_无人驾驶编程初体验
weixin_31509753的博客
02-06 1300
1月5日星期六,中关村创业大街创业博物馆学生开放日,一场有趣的无人驾驶编程科普活动悄然展开。在短暂的时间里,通过项目式学习的方式,从无人驾驶的原理到编程再到实践,让这些零基础的孩子深入学习了无人驾驶相关知识,体验到人工智能的乐趣。这次短暂的科普体验活动,通过无人驾驶项目(主题)应用,结合完整的赛车组装、调试、编程过程,学习如何通过编程实现图像识别,在无人驾驶应用领域和执行器电路的综合编程应用。主讲...
智能小车知识点总结
勾栏听曲_0的博客
06-11 3703
STM23控制智能小车的基本知识,涉及红外,中断,定时,蓝牙等
基于STM32的智能循迹避障小车实验(小车运动部分)
最新发布
only_print的博客
04-11 5503
这个实验是关于智能小车的实验,现在的想法就是先做出一个循迹和避障功能,后续可能会再添加一些其他的模块。
基于树莓派4B的智能无人巡逻小车设计
cangzhexingxing的博客
06-03 1万+
本实验设计的场景是智能警用无人巡逻小车,可以自动巡线,精准避障,遇到障碍物时实现S型绕弯,同时闪烁LED灯提醒,智能检测障碍物是车牌还是无关障碍,完成识别及做出相应的动作,完成信息发送,车牌存档等等操作,再实现精准避开车牌,若遇到四方都有障碍则原路返回,巡线直至检测到停车线,停车入库等内容。......
一步步制作Arduino智能小车
热门推荐
m0_67565756的博客
07-06 1万+
小白第一次实战
循迹小车具体做法与经验分享
To_be_NO_one的博客
04-25 9846
简单介绍 刚开学时做了一个循迹小车,因为第一次做小项目,没有经验,走了很多弯路,前前后后花费了大概一个月的时间。被小车折磨了一个月后,我学到了很多东西,体会很深,很想写下来与大家分享。主要献给和我一样第一次做项目的朋友们,希望对你们有用。 接下来我分享一下我这个小车的做法和经验。整个过程主要分为查资料、购买原件、组装、编程序、调试小车五步。在这篇博客中我将具体介绍这五步,让你看完后对循迹小车有大体...
Led 模块的使用
竹影卿心的博客
07-07 1002
一个实例,让led灯闪烁起来。 首先,使用杜邦线将组件与主机连接起来,V 接5v电源,G 接GND,S接GPIO接口。 这里GPIO接口选择了33。 红色接5v电源,黄色接GPIO33,黑色GND 创建led.py,代码如下 执行py文件 效果,红色灯闪烁,30秒后停止 ...
智能小车与超声波模块编程实践教程
资源摘要信息:"本资源为超声波小车相关资料的压缩文件,包含有关智能小车以及超声波模块的程序代码,为学习和实验目的而整理。" 关键词:超声波小车、超声波模块、51程序 知识点: 1. 超声波小车简介: 超声波...
写文章

热门文章

  • Android Kotlin的Class、反射、泛型 16798
  • Android listView同时展示多种不同数据的item 12289
  • fragment重叠问题(add hide show方式) 7599
  • Android OTG 读取usb设备文件并上传(问题解决) 6440
  • GPS定位系统(三)——Java后端 5463

分类专栏

  • devOps 9篇
  • jenkins 1篇
  • java 3篇
  • 正能量 1篇
  • blockly 2篇
  • GPS定位系统 8篇
  • Android 7篇
  • 微信小程序

最新评论

  • kafka3.5.1(raft版本 sasl认证)集群docker部署

    qq_40094384: KAFKA_CFG_LISTENERS 跟KAFKA_LISTENERS 这两个是不是都可以生效呢

  • kafka3.5.1(raft版本 sasl认证)集群docker部署

    Jafir: 你好 2181是zookeeper的端口 现在最新版已经不使用zk了 所以自然也不用 zk的数据信息来实现可视化监控 可视化监控 部署其他的组件吧 比如文内 kafka-ui 或者其他 找找看

  • kafka3.5.1(raft版本 sasl认证)集群docker部署

    qq_40094384: 你好 kraft模式 怎么暴漏2181端口呢(efak可视化界面需要连接这个端口)

  • 曾经的我给初入大学的你们

    dreamer'~: 引用「学会对知识的回顾和总结。最有效最简单的办法,就是在每次读完文章、博客,学习一种新的知识之后,都要回顾」 【回顾的力量】大佬说的很有用表情包,除了读别人的博客以外,自己写博客感觉也是一样的。 比如自己写完某篇博客文章后,可以在文末加个小结,提炼出整篇文章的重点和精华所在。这既是回顾总结,方便日后自己再次翻阅这篇文章时,能快速抓住重点;同时也能给自己加深印象,巩固学习成果。

  • 写博客多平台发布简书、掘金、CSDN

    shandianchengzi: 散了,bloghelper已经删库了,gitee也已经明令禁止用作图床了

最新文章

  • k8s+nacos无缝发布方案
  • devOps系列(八)efk+prometheus+grafana日志监控和告警
  • devOps系列(七)grafana+prometheus监控告警
2024年11篇
2021年8篇
2020年25篇
2018年10篇
2015年4篇

目录

目录

评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43元 前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值

深圳坪山网站建设公司罗源网站seo优化鹰潭网站优化推广蓟县网站优化哪家效果好清丰县网站的优化济阳济南网站建设优化东莞网站优化哪家好eo网站优化网站推广优化信息最新网站排名优化方法平顶山网站关键词优化代理武汉网站怎么优化长尾词江苏网站优化推荐复兴区网站优化多少钱海口网站优化价值优化网站链接的心得曹县个性化网站优化推广哪家好石家庄专业网站优化如何做肥西网站seo优化亚东网站优化专业网站优化外包费用丹凤县网站优化邳州网站优化制作莱阳网站优化代理商宁波做网站优化哪家好鹤壁网站排名优化找哪家江苏企业网站优化公司电子网站优化重庆专业网站优化平台南宁哪里有网站seo优化鄂城区网站做优化代理加盟香港通过《维护国家安全条例》两大学生合买彩票中奖一人不认账让美丽中国“从细节出发”19岁小伙救下5人后溺亡 多方发声卫健委通报少年有偿捐血浆16次猝死汪小菲曝离婚始末何赛飞追着代拍打雅江山火三名扑火人员牺牲系谣言男子被猫抓伤后确诊“猫抓病”周杰伦一审败诉网易中国拥有亿元资产的家庭达13.3万户315晚会后胖东来又人满为患了高校汽车撞人致3死16伤 司机系学生张家界的山上“长”满了韩国人?张立群任西安交通大学校长手机成瘾是影响睡眠质量重要因素网友洛杉矶偶遇贾玲“重生之我在北大当嫡校长”单亲妈妈陷入热恋 14岁儿子报警倪萍分享减重40斤方法杨倩无缘巴黎奥运考生莫言也上北大硕士复试名单了许家印被限制高消费奥巴马现身唐宁街 黑色着装引猜测专访95后高颜值猪保姆男孩8年未见母亲被告知被遗忘七年后宇文玥被薅头发捞上岸郑州一火锅店爆改成麻辣烫店西双版纳热带植物园回应蜉蝣大爆发沉迷短剧的人就像掉进了杀猪盘当地回应沈阳致3死车祸车主疑毒驾开除党籍5年后 原水城县长再被查凯特王妃现身!外出购物视频曝光初中生遭15人围殴自卫刺伤3人判无罪事业单位女子向同事水杯投不明物质男子被流浪猫绊倒 投喂者赔24万外国人感慨凌晨的中国很安全路边卖淀粉肠阿姨主动出示声明书胖东来员工每周单休无小长假王树国卸任西安交大校长 师生送别小米汽车超级工厂正式揭幕黑马情侣提车了妈妈回应孩子在校撞护栏坠楼校方回应护栏损坏小学生课间坠楼房客欠租失踪 房东直发愁专家建议不必谈骨泥色变老人退休金被冒领16年 金额超20万西藏招商引资投资者子女可当地高考特朗普无法缴纳4.54亿美元罚金浙江一高校内汽车冲撞行人 多人受伤

深圳坪山网站建设公司 XML地图 TXT地图 虚拟主机 SEO 网站制作 网站优化