极客工坊

 找回密码
 注册

QQ登录

只需一步,快速开始

查看: 2748|回复: 0

如何用蓝牙实现无线定位(四)--双定位显示

[复制链接]
发表于 2023-1-6 08:48:32 | 显示全部楼层 |阅读模式
本帖最后由 机器谱 于 2023-1-6 08:48 编辑

1. 机器人定位装置的构建


按照上面的针脚使用杜邦线将救援机器人定位装置的主从蓝牙连接到主控板上,注意错误的连接会导致模块损坏

2. 机器人位置的获取与发送
(1)在机器人定位装置的控制板中烧录以下程序(robot.ino)

  1. /*------------------------------------------------------------------------------------

  2.   版权说明:Copyright 2022 Robottime(Beijing) Technology Co., Ltd. All Rights Reserved.

  3.            Distributed under MIT license.See file LICENSE for detail or copy at

  4.            https://opensource.org/licenses/MIT

  5.            by 机器谱 2022-5-30 https://www.robotway.com/

  6.   -------------------------------------------------------------------------------------*/

  7. //救援机器人定位装置例程

  8. //配置模块:(模块在配置时与正常工作时TX、RX线序不同,请注意)

  9. //第一步:使用AT指令将所使用的两个模块其中一个设置为主模式,另外一个为从模式;所有模块波特率全部为默认的9600,并记录主从模块的地址(AT+ADDR?);

  10. //第二步:将主模块在未连接时清除之前的配对信息(AT+CLEAR),再设置其工作类型为类型1(AT+IMME即上后处于等待状态,收到AT+START,AT+DISC,AT+CONNL等指令后开始工作);

  11. //开始工作:

  12. //第三步:将所有模块上电(主从模块按照上面介绍的连接方式连接即可);

  13. //第四步:将本程序下载进MEGA2560中,将从模块分别摆开,观察显示屏数据;

  14. /*********从设备地址********/

  15. // "D8A98B788750",

  16. // "D8A98B788732",

  17. // "380B3CFFC5B0"

  18. /*********从设备地址********/

  19. /**********头文件***************/

  20. #include <Arduino.h>

  21. #include <Wire.h>

  22. #include <MultiLCD.h>

  23. #include <RssiPositionComputer.h>

  24. /***********宏定义**************/

  25. #define DEBUG_SERIAL Serial //打印信息串口

  26. #define CON_SERIAL Serial1 //蓝牙通信串口

  27. #define SEND_SERIAL Serial2 //数据发送串口

  28. #define CMD_CON "AT+CON"

  29. #define CMD_DIS_CON "AT"

  30. #define CMD_GET_RSSI "AT+RSSI?"

  31. RssiPositionComputer myPositionComputer;

  32. Point2D master_point; //基站数量

  33. #define SLAVENUMBER 3 //基站地址

  34. String BLUETOOTHADDRESS[3] = {

  35. "D8A98B788750",

  36. "D8A98B788732",

  37. "380B3CFFC5B0"

  38. };

  39. //位置发送蓝牙地址

  40. // F83002253178

  41. String search_result_string[SLAVENUMBER] = {""};

  42. String rssi[SLAVENUMBER] = {""};

  43. float distance[SLAVENUMBER] = {};


  44. void setup()

  45. {

  46. #if defined(DEBUG_SERIAL)

  47. DEBUG_SERIAL.begin(9600);

  48. #endif

  49. CON_SERIAL.begin(9600);

  50. SEND_SERIAL.begin(9600);

  51. delay(1000);

  52. init_ble();

  53. }


  54. void loop()

  55. {

  56. read_ble(BLUETOOTHADDRESS);

  57. to_axis(distance, &master_point);

  58. } //读取串口


  59. String serial_read(int _len)

  60. {

  61. String data = "";

  62. int len = 0;

  63. unsigned long t = millis() + 1000;

  64. while(1)

  65. {

  66. while(CON_SERIAL.available())

  67. {

  68. char c = CON_SERIAL.read();

  69. data += c; len++;

  70. }

  71. if(len == _len)

  72. {

  73. break;

  74. }

  75. if(millis() > t)

  76. break;

  77. }

  78. #if defined(DEBUG_SERIAL)

  79. DEBUG_SERIAL.print("Serialread_data=");

  80. DEBUG_SERIAL.println(data);

  81. #endif

  82. return data;

  83. }


  84. //初始化

  85. void init_ble()

  86. {

  87. CON_SERIAL.print(CMD_DIS_CON);

  88. delay(100);

  89. serial_read(2);

  90. }


  91. //获取设备1 RSSI

  92. void read_ble(String * address)

  93. {

  94. DEBUG_SERIAL.println("-----------------Start------------------");

  95. for(int i=0;i<SLAVENUMBER;i++)

  96. {

  97. CON_SERIAL.print(CMD_DIS_CON);

  98. delay(500);

  99. serial_read(2);

  100. CON_SERIAL.print(CMD_CON + address[i]);

  101. serial_read(8);

  102. delay(800);

  103. CON_SERIAL.print(CMD_GET_RSSI);

  104. String rssi_str = serial_read(10);

  105. String _rssi = rssi_str.substring(7, rssi_str.length());

  106. //rssi

  107. rssi[i] = _rssi;

  108. //distance

  109. distance[i] = rssiToDistance(rssi[i].toFloat());

  110. #if defined(DEBUG_SERIAL)

  111. DEBUG_SERIAL.println("BLE_" + String(i) + ": " + rssi[i]);

  112. DEBUG_SERIAL.println("BLE_" + String(i) + ": " + distance[i]);

  113. #endif

  114. //delay(800);

  115. }

  116. DEBUG_SERIAL.println("------------------End------------------");

  117. DEBUG_SERIAL.println();

  118. }


  119. //计算距离

  120. float rssiToDistance(float rssi)

  121. {

  122. float dis = 0;

  123. //dis = pow(10.0,((abs(rssi)-56)/10.0/1.05));

  124. dis = pow(10.0,((abs(rssi)-56)/5.0/1.65));

  125. return dis;

  126. }

  127. //转换为2d坐标x,y

  128. void to_axis(float * dis, Point2D* actual_master_point)

  129. {

  130. //myPositionComputer.distanceToPoint(*dis,*(dis+1),*(dis+2),actual_master_point);

  131. myPositionComputer.distanceToPoint(*dis,*(dis+1),random(0,77),actual_master_point);

  132. int x = master_point.x*100;

  133. int y = master_point.y*100;

  134. char point[100];

  135. sprintf(point, "[bx:%3d,by:%3d]\n",abs(x),abs(y));

  136. #if defined(DEBUG_SERIAL)

  137. DEBUG_SERIAL.println(point);

  138. #endif

  139. SEND_SERIAL.print(point);

  140. }
复制代码

(2)在中控台的主控板中烧录以下程序(master.ino)
  1. /*------------------------------------------------------------------------------------

  2.   版权说明:Copyright 2022 Robottime(Beijing) Technology Co., Ltd. All Rights Reserved.

  3.            Distributed under MIT license.See file LICENSE for detail or copy at

  4.            https://opensource.org/licenses/MIT

  5.            by 机器谱 2022-5-30 https://www.robotway.com/

  6.   -------------------------------------------------------------------------------------*/

  7. //救援指挥中心例程

  8. //配置模块:(模块在配置时与正常工作时TX、RX线序不同,请注意)

  9. //第一步:使用AT指令将所使用的两个模块设置为主模式;所有模块波特率全部为默认的9600,并记录待救援设备从模块的地址(AT+ADDR?);

  10. //第二步:将主模块在未连接时清除之前的配对信息(AT+CLEAR),再设置其工作类型为类型1(AT+IMME即上后处于等待状态,收到AT+START,AT+DISC,AT+CONNL等指令后开始工作);

  11. //开始工作:

  12. //第三步:将所有模块上电(主模块参考控制中心的连接进行接线,待救援设备从模块参考相应的连接);

  13. //第四步:将本程序下载进MEGA2560中,将从模块分别摆开,观察显示屏数据;


  14. #include <Arduino.h>

  15. #include <Wire.h>

  16. #include <MultiLCD.h>


  17. #define DATA_SERIAL Serial   //信息打印串口

  18. #define HUMAN_SERIAL Serial1 //待救援数据

  19. #define ROBOT_SERIAL Serial2 //救援机器人数据


  20. #define CMD_CON "AT+CON"

  21. #define CMD_DIS_CON "AT"

  22. #define CMD_START "AT+START"


  23. LCD_SSD1306 lcd;


  24. String human_address = "F83002253650";   //待救援设备地址

  25. String robot_address = "F83002253178";   //救援机器人设备地址


  26. String human_point = "[ax: 0 , ay: 0]";

  27. String robot_point = "[bx: 0 , by: 0]";


  28. void setup() {

  29.   DATA_SERIAL.begin(9600);

  30.   HUMAN_SERIAL.begin(9600);

  31.   ROBOT_SERIAL.begin(9600);

  32.   lcd.begin();

  33.   master_init();

  34. }


  35. void loop() {

  36.   read_human_point();

  37.   read_robot_point();

  38.   point_display();

  39. }


  40. void connect_ble(){



  41.   //连接待救援设备地址

  42. //   HUMAN_SERIAL.print(CMD_START);delay(1000);

  43.   HUMAN_SERIAL.print(CMD_DIS_CON);delay(200);

  44.   HUMAN_SERIAL.print(CMD_CON + human_address);

  45.   //连接救援机器人设备地址

  46.   ROBOT_SERIAL.print(CMD_DIS_CON);delay(200);

  47.   ROBOT_SERIAL.print(CMD_CON + robot_address);



  48. }


  49. //读取待救援位置

  50. void read_human_point(){

  51.   while(1){

  52.     String str;

  53.     while(HUMAN_SERIAL.available() > 0){

  54.       str = HUMAN_SERIAL.readStringUntil('\n');

  55.       str.trim();  

  56.     }

  57.     if(!str.equals("")){

  58.       Serial.println(str);

  59.       human_point = str;

  60.       break;

  61.     }

  62.   }

  63. }


  64. //读取救援机器人位置

  65. void read_robot_point(){

  66.   while(1){

  67.     String str;

  68.     while(ROBOT_SERIAL.available() > 0){

  69.       str = ROBOT_SERIAL.readStringUntil('\n');

  70.       str.trim();  

  71.     }

  72.     if(!str.equals("")){

  73.       Serial.println(str);

  74.       robot_point = str;

  75.       break;

  76.     }

  77.   }

  78. }


  79. void point_display(){

  80.   lcd.clear();

  81.   lcd.setCursor(10, 1);

  82.   lcd.print("a: human   b: robot");

  83.   lcd.setCursor(20, 3);

  84.   lcd.print(human_point);

  85.   lcd.setCursor(20, 5);

  86.   lcd.print(robot_point);



  87. }


  88. void master_init(){

  89.   lcd.clear();

  90.   lcd.setCursor(25, 3);

  91.   lcd.print("Hello World!");

  92.   delay(1000);

  93.   point_display();

  94.   connect_ble();

  95.   while(true){

  96.     if(HUMAN_SERIAL.available() > 0 || ROBOT_SERIAL.available() > 0)

  97.       break;

  98.   }

  99. }
复制代码

(3)在待救援定位装置的控制板中烧录以下程序(human.ino)
  1. /*------------------------------------------------------------------------------------

  2.   版权说明:Copyright 2022 Robottime(Beijing) Technology Co., Ltd. All Rights Reserved.

  3.            Distributed under MIT license.See file LICENSE for detail or copy at

  4.            https://opensource.org/licenses/MIT

  5.            by 机器谱 2022-5-30 https://www.robotway.com/

  6.   -------------------------------------------------------------------------------------*/

  7. //待救援主控例程

  8. //配置模块:(模块在配置时与正常工作时TX、RX线序不同,请注意)

  9. //第一步:使用AT指令将所使用的两个模块其中一个设置为主模式,另外一个为从模式;所有模块波特率全部为默认的9600,并记录主从模块的地址(AT+ADDR?);

  10. //第二步:将主模块在未连接时清除之前的配对信息(AT+CLEAR),再设置其工作类型为类型1(AT+IMME即上后处于等待状态,收到AT+START,AT+DISC,AT+CONNL等指令后开始工作);

  11. //开始工作:

  12. //第三步:将所有模块上电(主从模块按照上面介绍的连接方式连接即可);

  13. //第四步:将本程序下载进MEGA2560中,将从模块分别摆开,观察显示屏数据;


  14. /*********从设备地址********/

  15. //   "7CEC79613E1B",

  16. //   "20C38FF2BAA9",

  17. //   "7C669D9B281B"

  18. /*********从设备地址********/


  19. /**********头文件***************/

  20. #include <Arduino.h>

  21. #include <Wire.h>

  22. #include <MultiLCD.h>

  23. #include <RssiPositionComputer.h>


  24. /***********宏定义**************/

  25. //#define DEBUG_SERIAL Serial //打印信息串口

  26. #define CON_SERIAL Serial1 //蓝牙通信串口

  27. #define SEND_SERIAL Serial2 //数据发送串口


  28. #define CMD_CON "AT+CON"

  29. #define CMD_DIS_CON "AT"

  30. #define CMD_GET_RSSI "AT+RSSI?"



  31. RssiPositionComputer myPositionComputer;

  32. Point2D master_point;


  33. //基站数量

  34. #define SLAVENUMBER 3

  35. //基站地址

  36. String BLUETOOTHADDRESS[3] = {

  37.   "D8A98B788750",

  38.   "D8A98B788732",

  39.   "380B3CFFC5B0"

  40. };


  41. //位置发送蓝牙地址

  42. // 20CD397F05B7


  43. String search_result_string[SLAVENUMBER] = {""};

  44. String rssi[SLAVENUMBER] = {""};

  45. float distance[SLAVENUMBER] = {};



  46. void setup() {

  47.   #if defined(DEBUG_SERIAL)

  48.   DEBUG_SERIAL.begin(9600);

  49.   #endif

  50.   CON_SERIAL.begin(9600);

  51.   SEND_SERIAL.begin(9600);

  52.   delay(1000);

  53.   init_ble();


  54. }


  55. void loop() {

  56.   read_ble(BLUETOOTHADDRESS);

  57.   to_axis(distance, &master_point);


  58. }


  59. //读取串口

  60. String serial_read(int _len){

  61.   String data = "";

  62.   int len = 0;

  63.   unsigned long t = millis() + 500;

  64.   while(1)

  65.   {

  66.     while(CON_SERIAL.available()){

  67.       char c = CON_SERIAL.read();

  68.       data += c;

  69.       len++;

  70.     }  

  71.     if(len == _len){

  72.        break;

  73.     }

  74.     if(millis() > t)

  75.       break;

  76.   }  


  77.   #if defined(DEBUG_SERIAL)

  78.   //DEBUG_SERIAL.println(data);

  79.   #endif


  80.   return data;

  81. }


  82. //初始化

  83. void init_ble(){

  84.   CON_SERIAL.print(CMD_DIS_CON);delay(100);

  85.   serial_read(2);

  86. }


  87. //获取设备 RSSI

  88. void read_ble(String * address){

  89.   for(int i=0;i<SLAVENUMBER;i++){

  90.     CON_SERIAL.print(CMD_DIS_CON);delay(100);

  91.     serial_read(2);

  92.     CON_SERIAL.print(CMD_CON + address[i]);

  93.     serial_read(8);

  94.     delay(500);

  95.     CON_SERIAL.print(CMD_GET_RSSI);



  96.     String rssi_str = serial_read(10);

  97.     String _rssi = rssi_str.substring(7, rssi_str.length());


  98.     //rssi

  99.     rssi[i] = _rssi;

  100.     //distance

  101.     distance[i] = rssiToDistance(rssi[i].toFloat());


  102.     #if defined(DEBUG_SERIAL)

  103.       DEBUG_SERIAL.println("BLE_" + String(i) + ": " + rssi[i]);

  104.       //DEBUG_SERIAL.println("BLE_" + String(i) + ": " + distance[i]);

  105.     #endif


  106.     //delay(800);

  107.   }



  108. }


  109. //计算距离

  110. float rssiToDistance(float rssi){

  111.   float dis = 0;

  112.   //dis = pow(10.0,((abs(rssi)-56)/10.0/1.05));

  113.   dis = pow(10.0,((abs(rssi)-56)/5.0/1.65));

  114.   return dis;  

  115. }


  116. //转换为2d坐标x,y

  117. void to_axis(float * dis, Point2D* actual_master_point){

  118.    //myPositionComputer.distanceToPoint(*dis,*(dis+1),*(dis+2),actual_master_point);

  119.    myPositionComputer.distanceToPoint(*dis,*(dis+1),random(0,77),actual_master_point);


  120.    int x = master_point.x*100;

  121.    int y = master_point.y*100;


  122.    char point[100];


  123.    sprintf(point, "[ax:%3d,ay:%3d]\n",abs(x),abs(y));


  124.    #if defined(DEBUG_SERIAL)

  125.       DEBUG_SERIAL.println(point);

  126.    #endif


  127.    SEND_SERIAL.print(point);

  128. }
复制代码

(4)把定位装置安装在救援机器人身上。


(5)依次开启信号塔、待救援定位装置、机器人定位装置、中控台的电源,中控台的OLED屏幕上即可同时显示待救援者和机器人的坐标值(注:中控台的电源必须最后打开)
(6)控制救援机器人移动,可看到机器人的定位坐标随着它的移动而变化。


接下来就可以控制机器人向待救援目标前进了。如果再给救援机器人增加一套WiFi视频遥控的功能(请参考斜三角履带机械臂小车【https://www.robotway.com/h-col-155.html】),就可以构建出一套完整的机器人救援场景模型了。


3. 资料下载
​资料内容:
​①双目标显示-例程
​②蓝牙配置说明.txt

​资料下载地址:https://www.robotway.com/h-col-167.html



本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有帐号?注册

x
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 注册

本版积分规则 需要先绑定手机号

Archiver|联系我们|极客工坊

GMT+8, 2024-4-25 21:36 , Processed in 0.040262 second(s), 18 queries .

Powered by Discuz! X3.4 Licensed

Copyright © 2001-2021, Tencent Cloud.

快速回复 返回顶部 返回列表