当前位置: 首页 > news >正文

【第六课】ESP32-S3 BLE 与 WiFi 一体化通信示例

🚀 ESP32-S3 BLE 与 WiFi 一体化通信示例

本项目展示了如何使用 ESP32-S3 实现 BLE 通信接收 WiFi 配置信息,并尝试连接 WiFi,同时通过 BLE 向客户端发送心率数据和 WiFi 连接状态。


📌 一、功能概述

  • 使用 BLE 广播设备
  • 提供 BLE 特性用于接收 WiFi SSID 和密码
  • 自动尝试连接 WiFi,并通过 BLE 返回连接状态
  • 周期性发送模拟心率数据(用于调试展示)

⚙️ 二、核心技术与库

  • BLEDevice, BLEUtils, BLEServer: ESP32 BLE API
  • WiFi.h: 管理 WiFi 连接
  • 使用 BLE 特性传输配置参数(如 WiFi)
  • 使用 BLE notify 功能主动推送状态

🧠 三、UUID 设置

功能描述UUID说明
传感器服务180D心率服务
心率数据特性2A37notify 心率
WiFi服务180F自定义为 WiFi 服务
WiFi配置接收特性2A38write,接收SSID+密码
WiFi连接状态特性2A39notify,返回连接结果

🔧 四、代码结构概述

1. 初始化 BLE 服务器与服务

pServer = BLEDevice::createServer();
pService = pServer->createService(SENSOR_SERVICE_UUID);

2. 添加 BLE 特性(心率、WiFi配置、连接状态)

pHeartRateCharacteristic = pService->createCharacteristic(...);
pWiFiConfigReceiveCharacteristic = pService->createCharacteristic(...);
pWiFiConnectionStatusCharacteristic = pService->createCharacteristic(...);

3. 特性回调处理 WiFi 数据

void onWrite(BLECharacteristic *pCharacteristic) {receivedData = pCharacteristic->getValue().c_str();// 提取 SSID 和密码并连接connectToWiFi();
}

4. WiFi 连接尝试与状态反馈

WiFi.begin(wifiSSID.c_str(), wifiPassword.c_str());
if (WiFi.status() == WL_CONNECTED) {pWiFiConnectionStatusCharacteristic->setValue("连接成功");
} else {pWiFiConnectionStatusCharacteristic->setValue("连接失败");
}
pWiFiConnectionStatusCharacteristic->notify();

5. 循环中周期性发送心率数据

if (deviceConnected) {heartRate = (heartRate + 1) % 100;pHeartRateCharacteristic->setValue(...);pHeartRateCharacteristic->notify();
}

🔍 五、BLE 客户端格式(发送 WiFi)

客户端通过如下格式写入数据至 2A38 特性:

SSID;PASSWORD

例如:

MyHomeWiFi;my_password123

📝 六、调试输出参考

BLE广播已启动
设备已连接
接收到的数据:MyWiFi;MyPass
WiFi已连接
IP地址: 192.168.1.88

🧩 七、应用场景

  • 终端设备初始配置
  • 无需物理界面完成联网
  • BLE App 快速调试嵌入式设备

📁 八、完整代码与资源

#include <BLEDevice.h>
#include <BLEUtils.h>
#include <BLEServer.h>
#include <BLE2902.h>
#include <WiFi.h>// 传感器服务UUID
#define SENSOR_SERVICE_UUID        "180D"  // 传感器服务的UUID// 心率测量特性UUID
#define HEART_RATE_CHARACTERISTIC_UUID  "2A37"  // 其他传感器特性UUID(此处可以扩展,如添加温湿度传感器等)
// #define TEMPERATURE_CHARACTERISTIC_UUID "XXXX"  
// #define HUMIDITY_CHARACTERISTIC_UUID    "YYYY"// WiFi服务UUID
#define WIFI_SERVICE_UUID           "180F"  // WiFi服务UUID,假设为此值// WiFi配置接收UUID(接收WiFi信息)
#define WIFI_CREDENTIAL_RECEIVE_UUID "2A38"  // WiFi连接状态UUID(发送WiFi连接状态)
#define WIFI_CONNECTION_STATUS_UUID "2A39"  BLECharacteristic *pHeartRateCharacteristic;
BLECharacteristic *pWiFiConfigReceiveCharacteristic;
BLECharacteristic *pWiFiConnectionStatusCharacteristic;
BLEServer *pServer;
bool deviceConnected = false;
String receivedData = "";  // 存储接收到的数据// WiFi账号和密码
String wifiSSID = "";
String wifiPassword = "";// 提前声明connectToWiFi函数
void connectToWiFi();// 设备连接/断开回调
class MyServerCallbacks : public BLEServerCallbacks {void onConnect(BLEServer* pServer) {deviceConnected = true;Serial.println("设备已连接");}void onDisconnect(BLEServer* pServer) {deviceConnected = false;Serial.println("设备已断开连接");BLEAdvertising *pAdvertising = pServer->getAdvertising();pAdvertising->start();Serial.println("BLE广播重新启动");}
};// 接收数据的特性回调
class MyCharacteristicCallbacks : public BLECharacteristicCallbacks {void onWrite(BLECharacteristic *pCharacteristic) {receivedData = pCharacteristic->getValue().c_str();Serial.print("接收到的数据:");Serial.println(receivedData);// 分离SSID和密码int separatorIndex = receivedData.indexOf(';');if (separatorIndex != -1) {wifiSSID = receivedData.substring(0, separatorIndex);wifiPassword = receivedData.substring(separatorIndex + 1);Serial.print("WiFi SSID: ");Serial.println(wifiSSID);Serial.print("WiFi Password: ");Serial.println(wifiPassword);// 尝试连接WiFiconnectToWiFi();}}
};void connectToWiFi() {Serial.println("尝试连接到WiFi...");WiFi.begin(wifiSSID.c_str(), wifiPassword.c_str());int retryCount = 0;while (WiFi.status() != WL_CONNECTED && retryCount < 20) {delay(500);Serial.print(".");retryCount++;}if (WiFi.status() == WL_CONNECTED) {Serial.println("\nWiFi已连接");Serial.print("IP地址: ");Serial.println(WiFi.localIP());// 连接成功,发送状态给APPpWiFiConnectionStatusCharacteristic->setValue("连接成功");pWiFiConnectionStatusCharacteristic->notify();} else {Serial.println("\nWiFi连接失败,请检查SSID和密码");// 连接失败,发送状态给APPpWiFiConnectionStatusCharacteristic->setValue("连接失败");pWiFiConnectionStatusCharacteristic->notify();}
}void setup() {Serial.begin(115200);BLEDevice::init("智能跑步机BLE调试");// 创建BLE服务器pServer = BLEDevice::createServer();pServer->setCallbacks(new MyServerCallbacks());// 创建传感器服务BLEService *pService = pServer->createService(SENSOR_SERVICE_UUID);// 创建心率测量特性pHeartRateCharacteristic = pService->createCharacteristic(HEART_RATE_CHARACTERISTIC_UUID,BLECharacteristic::PROPERTY_READ |BLECharacteristic::PROPERTY_NOTIFY);pHeartRateCharacteristic->setValue("HeartRate: 75 bpm");pHeartRateCharacteristic->addDescriptor(new BLE2902());// 创建WiFi配置接收特性pWiFiConfigReceiveCharacteristic = pService->createCharacteristic(WIFI_CREDENTIAL_RECEIVE_UUID,BLECharacteristic::PROPERTY_WRITE);pWiFiConfigReceiveCharacteristic->setCallbacks(new MyCharacteristicCallbacks());// 创建WiFi连接状态通知特性pWiFiConnectionStatusCharacteristic = pService->createCharacteristic(WIFI_CONNECTION_STATUS_UUID,BLECharacteristic::PROPERTY_NOTIFY);// 启动服务pService->start();// 启动广播BLEAdvertising *pAdvertising = pServer->getAdvertising();//pAdvertising->addServiceUUID(SENSOR_SERVICE_UUID);  // 只广播自定义的服务UUID——设置广播信息,确保只广播自定义的服务UUIDpAdvertising->setScanResponse(true);pAdvertising->start();Serial.println("BLE广播已启动");
}void loop() {static int heartRate = 75;if (deviceConnected) {heartRate = (heartRate + 1) % 100;String heartRateStr = "HeartRate: " + String(heartRate) + " bpm";pHeartRateCharacteristic->setValue(heartRateStr.c_str());pHeartRateCharacteristic->notify();Serial.println(heartRateStr);}delay(1000);
}

相关文章:

  • 从0开始学linux韦东山教程第三章问题小结(4)
  • 内存分配器ptmalloc2、tcmalloc、jemalloc,结构设计、内存分配过程详解
  • fiftyone-数据库配置和config与app_config配置文件
  • 耐压击穿测试的原理、流程及应用领域
  • SymPy | 如何提取指定项的系数
  • LeetCode 820 单词的压缩编码题解
  • 笔记本电脑打开网页很慢,一查ip地址网段不对怎么处理
  • 数学建模初等模型应用
  • 影刀RPA网页自动化总结
  • TCP 三次握手建立连接详解
  • 【PostgreSQL数据分析实战:从数据清洗到可视化全流程】附录-B. 错误代码与解决方案
  • AWS技术助力企业满足GDPR合规要求
  • MATLAB 中常用的微分函数介绍
  • 武汉科技大学人工智能与演化计算实验室许志伟课题组参加2025中国膜计算论坛
  • 网络运维过程中的常用命令
  • 安装npm:npm未随Node.js一起安装
  • 深度学习基础知识
  • 5月13日日记
  • 【行为型之策略模式】游戏开发实战——Unity灵活算法架构的核心实现策略
  • python三方库sqlalchemy
  • 特朗普促卡塔尔说服伊朗放弃核计划,伊朗总统:你来吓唬我们?
  • 光明日报:家长孩子共同“息屏”,也要保证高质量陪伴
  • 哪种“网红减肥法”比较靠谱?医学专家和运动专家共同解答
  • 河南省委常委会会议:坚持以案为鉴,深刻汲取教训
  • 今年前4个月上海对拉美国家进出口总值增长2%
  • 外交部:中方对美芬太尼反制仍然有效