esp32s3连接控制HC-08蓝牙设备

趁双十一买了一块esp32S3单片,尝试之后发现他的蓝牙只支持ble,我的机器人以前是使用手机控制的,我想借此机会,为他添加一个esp32S3的大脑。

查了一下资料,写了一个demo,记录一下代码:


#include "BLEDevice.h"
//#include "BLEScan.h"
//hc-08的ble需要至少两个uuid,一个是服务serviceUUID2,一个是读写charUUID,
//还有一个服务是扫描时用到serviceUUID1,这个服务是hc-08的广播服务,做为过滤蓝牙使用,这里也可以使用蓝牙的名字
// The remote service we wish to connect to.
//static BLEUUID serviceUUID("4fafc201-1fb5-459e-8fcc-c5c9c331914b");
static BLEUUID serviceUUID1("0000ffe1-0000-1000-8000-00805f9b34fb");
// The characteristic of the remote service we are interested in.
//static BLEUUID charUUID("0000ffe1-0000-1000-8000-00805f9b34fb");
static BLEUUID serviceUUID2("0000ffe0-0000-1000-8000-00805f9b34fb");
static BLEUUID charUUID("0000fff0-0000-1000-8000-00805f9b34fb");
static String name="RS-BLE";
/*
Advertised Device: Name: RS-BLE
, Address: 44:ee:14:4e:66:66, manufacturer data: 48432d30380000000000000000000000000000000000, serviceUUID: 0000ffe1-0000-1000-8000-00805f9b34fb, txPower: 4, rssi: -55 
*/

static boolean doConnect = false;
static boolean connected = false;
static boolean doScan = false;
static BLERemoteCharacteristic *pRemoteCharacteristic;
static BLEAdvertisedDevice *myDevice;

static void notifyCallback(BLERemoteCharacteristic *pBLERemoteCharacteristic, uint8_t *pData, size_t length, bool isNotify) {
  Serial.print("Notify callback for characteristic ");
  Serial.print(pBLERemoteCharacteristic->getUUID().toString().c_str());
  Serial.print(" of data length ");
  Serial.println(length);
  Serial.print("data: ");
  Serial.write(pData, length);
  Serial.println();
}

class MyClientCallback : public BLEClientCallbacks {
  void onConnect(BLEClient *pclient) {}

  void onDisconnect(BLEClient *pclient) {
    connected = false;
    Serial.println("onDisconnect");
  }
};

bool connectToServer() {
  Serial.print("Forming a connection to ");
  Serial.println(myDevice->getAddress().toString().c_str());

  BLEClient *pClient = BLEDevice::createClient();
  Serial.println(" - Created client");

  pClient->setClientCallbacks(new MyClientCallback());

  // Connect to the remove BLE Server.
  pClient->connect(myDevice);  // if you pass BLEAdvertisedDevice instead of address, it will be recognized type of peer device address (public or private)
  Serial.println(" - Connected to server");
  pClient->setMTU(517);  //set client to request maximum MTU from server (default is 23 otherwise)

  // Obtain a reference to the service we are after in the remote BLE server.
  BLERemoteService *pRemoteService = pClient->getService(serviceUUID2);
  if (pRemoteService == nullptr) {
    Serial.print("Failed to find our service UUID: ");
    Serial.println(serviceUUID2.toString().c_str());
    pClient->disconnect();
    return false;
  }
  Serial.println(" - Found our service");

  // Obtain a reference to the characteristic in the service of the remote BLE server.
  //std::map<std::string, BLERemoteCharacteristic *> bl = pRemoteService->getCharacteristics();

  pRemoteCharacteristic = pRemoteService->getCharacteristic(charUUID);
  if (pRemoteCharacteristic == nullptr) {
    Serial.print("Failed to find our characteristic UUID: ");
    Serial.println(charUUID.toString().c_str());
    pClient->disconnect();
    return false;
  }
  Serial.println(" - Found our characteristic");

  // Read the value of the characteristic.
  if (pRemoteCharacteristic->canRead()) {
    String value = pRemoteCharacteristic->readValue();
    Serial.print("The characteristic value was: ");
    Serial.println(value.c_str());
  }

  if (pRemoteCharacteristic->canNotify()) {
    pRemoteCharacteristic->registerForNotify(notifyCallback);
  }

  connected = true;
  return true;
}
/**
 * Scan for BLE servers and find the first one that advertises the service we are looking for.
 */
class MyAdvertisedDeviceCallbacks : public BLEAdvertisedDeviceCallbacks {
  /**
   * Called for each advertising BLE server.
   */
  void onResult(BLEAdvertisedDevice advertisedDevice) {
    Serial.print("BLE Advertised Device found: ");
    Serial.println(advertisedDevice.toString().c_str());

    // We have found a device, let us now see if it contains the service we are looking for.
    //if (advertisedDevice.haveName() && advertisedDevice.getName()=="RS-BLE") {
    if (advertisedDevice.haveServiceUUID() && advertisedDevice.isAdvertisingService(serviceUUID1)) {

      BLEDevice::getScan()->stop();
      myDevice = new BLEAdvertisedDevice(advertisedDevice);
      doConnect = true;
      doScan = true;

    }  // Found our server
  }  // onResult
};  // MyAdvertisedDeviceCallbacks


void setup() {
  Serial.begin(115200);
  Serial.println("Starting Arduino BLE Client application...");
  BLEDevice::init("");

  // Retrieve a Scanner and set the callback we want to use to be informed when we
  // have detected a new device.  Specify that we want active scanning and start the
  // scan to run for 5 seconds.
  BLEScan *pBLEScan = BLEDevice::getScan();
  pBLEScan->setAdvertisedDeviceCallbacks(new MyAdvertisedDeviceCallbacks());
  pBLEScan->setInterval(1349);
  pBLEScan->setWindow(449);
  pBLEScan->setActiveScan(true);
  pBLEScan->start(5, false);
}  // End of setup.

// This is the Arduino main loop function.
void loop() {

  // If the flag "doConnect" is true then we have scanned for and found the desired
  // BLE Server with which we wish to connect.  Now we connect to it.  Once we are
  // connected we set the connected flag to be true.
  if (doConnect == true) {
    if (connectToServer()) {
      Serial.println("We are now connected to the BLE Server.");
    } else {
      Serial.println("We have failed to connect to the server; there is nothing more we will do.");
    }
    doConnect = false;
  }

  // If we are connected to a peer BLE Server, update the characteristic each time we are reached
  // with the current time since boot.
  if (connected) {
    String newValue = "Time since boot: " + String(millis() / 1000);
    Serial.println("Setting new characteristic value to \"" + newValue + "\"");

    // Set the characteristic's value to be the array of bytes that is actually a string.
    pRemoteCharacteristic->writeValue(newValue.c_str(), newValue.length());

    uint8_t testArr[10] = {0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x11};
    pRemoteCharacteristic->writeValue(testArr, sizeof(testArr));  
  } else if (doScan) {
    BLEDevice::getScan()->start(0);  // this is just example to start scan after disconnect, most likely there is better way to do it in arduino
  }

  delay(1000);  // Delay a second between loops.
}  // End of loop

/* Based on Neil Kolban example for IDF: https://github.com/nkolban/esp32-snippets/blob/master/cpp_utils/tests/BLE%20Tests/SampleServer.cpp Ported to Arduino ESP32 by Evandro Copercini updates by chegewara */ #include <BLEDevice.h> #include <BLEUtils.h> #include <BLEServer.h> // See the following for generating UUIDs: // https://www.uuidgenerator.net/ #define SERVICE_UUID "4fafc201-1fb5-459e-8fcc-c5c9c331914b" #define CHARACTERISTIC_UUID "beb5483e-36e1-4688-b7f5-ea07361b26a8" void setup() { Serial.begin(115200); Serial.println("Starting BLE work!"); BLEDevice::init("Long name works now"); BLEServer *pServer = BLEDevice::createServer(); BLEService *pService = pServer->createService(SERVICE_UUID); BLECharacteristic *pCharacteristic = pService->createCharacteristic(CHARACTERISTIC_UUID, BLECharacteristic::PROPERTY_READ | BLECharacteristic::PROPERTY_WRITE); pCharacteristic->setValue("Hello World says Neil"); pService->start(); // BLEAdvertising *pAdvertising = pServer->getAdvertising(); // this still is working for backward compatibility BLEAdvertising *pAdvertising = BLEDevice::getAdvertising(); pAdvertising->addServiceUUID(SERVICE_UUID); pAdvertising->setScanResponse(true); pAdvertising->setMinPreferred(0x06); // functions that help with iPhone connections issue pAdvertising->setMinPreferred(0x12); BLEDevice::startAdvertising(); Serial.println("Characteristic defined! Now you can read it in your phone!"); } void loop() { // put your main code here, to run repeatedly: delay(2000); }
03-12
<think>好的,我现在需要帮助用户分析他们提供的ESP32 BLE服务器代码。首先,我需要仔细阅读用户给出的代码,理解每一部分的功能,然后按照用户的要求用中文进行解释,同时注意使用正确的LaTeX格式。 首先,用户提供的代码是一个基于Arduino框架的ESP32 BLE服务器示例。代码开头提到了Neil Kolban的例子,以及被移植到Arduino环境的过程。用户可能希望了解这段代码的结构和工作原理,或者是在使用过程中遇到了问题需要帮助解决。 接下来,我需要逐步分析代码的结构。代码开始包含了必要的BLE库头文件,然后定义了服务的UUID和特征的UUID。在setup函数中,初始化了串口通信,创建了BLE设备、服务器、服务和特征,并设置了特征的值为“Hello World says Neil”。之后,启动了服务并配置了BLE广播参数,最后开始广播。loop函数是空的,只有延迟,说明这个服务器在启动后主要处于待机状态。 用户可能关心的问题包括:UUID的作用、BLE服务器创建流程、特征属性的读写权限、广播设置的作用,以及如何在实际应用中使用这个服务器。此外,用户可能需要知道如何测试这段代码,比如使用手机应用来读取特征值。 我还需要注意用户提到的系统级指令,比如行内数学表达式使用$...$,但在这个问题中可能涉及较少,但如果有涉及公式的地方需要正确应用。例如,如果有提到数据传输速率或数学计算,可能需要使用LaTeX格式,但当前代码分析可能不涉及这些。 在解释代码时,需要确保结构清晰,分步骤说明每个部分的功能。例如,先说明头文件的作用,然后讲解UUID的定义,再逐步讲解setup中的初始化过程,服务的创建,特征的属性设置,广播配置,最后是loop的作用。 同时,用户可能想知道这段代码的实际应用场景,比如在IoT设备中作为外围设备提供数据,或者与其他设备进行通信。需要指出代码的关键点,比如服务UUID和特征UUID的重要性,以及如何修改这些值以适应不同的应用。 另外,可能需要提醒用户注意常见问题,比如UUID的生成方式,广播参数对设备连接性的影响,特别是针对iPhone的兼容性问题。代码中提到的setMinPreferred可能涉及到BLE广播间隔的设置,需要解释这些参数的作用。 最后,在总结部分需要归纳代码的功能,并给出测试建议,比如使用nRF Connect这样的BLE调试工具来验证服务是否正常运行,检查特征值是否正确,以及测试读写功能是否按预期工作。 需要确保回答准确无误,避免技术错误,比如BLE协议的相关细节是否正确,库函数的使用是否恰当。如果对某些部分不确定,应进行验证,例如检查BLEDevice::getAdvertising()和pServer->getAdvertising()的区别,确保解释正确。 总的来说,我需要将代码分解成各个部分,逐一解释,同时结合实际应用和可能的问题,帮助用户全面理解这段BLE服务器代码的工作原理和使用方法。</think>以下是对您提供的ESP32 BLE服务器代码的逐步解析: $\textcolor{blue}{【代码结构分析】}$ 1. **头文件引入** ```cpp #include <BLEDevice.h> #include <BLEUtils.h> #include <BLEServer.h> ``` 这三个库构成了ESP32的BLE开发基础: - `BLEDevice`:管理BLE设备初始化 - `BLEUtils`:提供BLE工具函数 - `BLEServer`:实现BLE服务器功能 2. **UUID定义** ```cpp #define SERVICE_UUID "4fafc201-1fb5-459e-8fcc-c5c9c331914b" #define CHARACTERISTIC_UUID "beb5483e-36e1-4688-b7f5-ea07361b26a8" ``` - 使用$UUID$(通用唯一标识符)标识服务和特征 - 可通过[在线生成器](https://www.uuidgenerator.net/)创建新的$UUID$ $\textcolor{blue}{【核心函数解析】}$ 3. **setup() 函数流程** ```cpp BLEDevice::init("Long name works now"); // 初始化BLE设备,设置设备名称 BLEServer *pServer = BLEDevice::createServer(); // 创建BLE服务器实例 BLEService *pService = pServer->createService(SERVICE_UUID); // 创建服务 // 创建可读写的特征 BLECharacteristic *pCharacteristic = pService->createCharacteristic( CHARACTERISTIC_UUID, BLECharacteristic::PROPERTY_READ | BLECharacteristic::PROPERTY_WRITE ); pCharacteristic->setValue("Hello World says Neil"); // 设置特征初始值 pService->start(); // 启动服务 ``` 4. **广播配置** ```cpp BLEAdvertising *pAdvertising = BLEDevice::getAdvertising(); pAdvertising->addServiceUUID(SERVICE_UUID); // 在广播数据中包含服务UUID pAdvertising->setScanResponse(true); // 允许扫描响应 pAdvertising->setMinPreferred(0x06); // 设置广播间隔参数 pAdvertising->setMinPreferred(0x12); // 优化iOS设备连接 BLEDevice::startAdvertising(); // 开始广播 ``` $\textcolor{blue}{【关键技术点】}$ - **特征属性**:通过`PROPERTY_READ | PROPERTY_WRITE`设置读写权限 - **广播参数**: - `0x06`对应96.25ms广播间隔 - `0x12`对应200ms扫描间隔 - **iOS兼容性**:`setMinPreferred()`专门用于改善iOS设备连接稳定性 $\textcolor{blue}{【实际应用场景】}$ 1. 作为智能家居设备的BLE接口 2. 实现手机与ESP32之间的双向数据传输 3. 构建低功耗传感器网络节点 $\textcolor{blue}{【测试建议】}$ 1. 使用手机APP(如nRF Connect)扫描设备 2. 验证服务UUID和特征UUID是否可见 3. 测试特征值的读取和写入操作 $\textcolor{red}{注意事项:}$ - 保持设备名称唯一性(避免与其他BLE设备冲突) - 特征值的最大长度限制为512字节 - 广播数据包总长度不超过31字节 该代码实现了一个基础BLE服务器架构,后续可通过添加回调函数实现更复杂的交互逻辑。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值