蓝牙 Mesh 组网指南:智能家居多设备联动

为什么选择蓝牙 Mesh?

说到智能家居组网,很多人第一反应是 WiFi 或者 Zigbee。但蓝牙 Mesh 其实是个被低估的选手。

它有几个明显优势:

  • 覆盖范围广:通过节点中继,信号可以穿墙过户,轻松覆盖 100+ 平米

  • 低功耗:设备待机功耗极低,电池供电的传感器能用几个月甚至一年

  • 成本低:蓝牙芯片价格亲民,ESP32 开发板几十块就能搞定

  • 手机直连:不需要额外网关,手机就能直接控制设备

如果你正在搭建智能家居系统,又不想花大价钱买成套方案,蓝牙 Mesh 是个很务实的选择。

硬件清单

设备型号数量单价总价
ESP32 开发板ESP32-WROOM-323 块¥25¥75
蓝牙 Mesh 模组JDY-642 个¥18¥36
温湿度传感器SHT301 个¥15¥15
继电器模块5V 单路2 个¥8¥16
USB 数据线Micro-USB3 根¥5¥15
面包板830 孔1 个¥12¥12
杜邦线公对公/母对母20 根¥8¥8
合计---¥177

所有配件都能在淘宝或立创商城买到,总价不到 200 块就能搭建一套完整的蓝牙 Mesh 网络。

蓝牙 Mesh 基础概念

在动手之前,先搞懂几个核心概念:

节点(Node)

每个加入 Mesh 网络的设备都是一个节点。节点可以:

  • 发送消息(比如温湿度传感器上报数据)

  • 接收消息(比如继电器接收开关指令)

  • 中继消息(帮助其他节点转发信号)

元素(Element)

一个节点可以包含多个元素。比如一个智能开关面板,可能有 3 个按键,每个按键就是一个独立的元素。

模型(Model)

模型定义了设备的功能。常见的有:

  • Generic OnOff:开关控制

  • Sensor:传感器数据上报

  • Light Lightness:灯光亮度调节

发布/订阅(Publish/Subscribe)

这是 Mesh 网络的核心机制:

  • 发布地址:设备向哪个地址发送消息

  • 订阅地址:设备监听哪个地址的消息

举个例子:温湿度传感器发布到地址 0xC001,继电器订阅 0xC001,这样传感器数据就能自动触发继电器动作。

环境搭建

1. 安装 ESP-IDF

ESP32 的官方开发框架是 ESP-IDF。我们用 Docker 方式安装,避免污染系统环境:

# 拉取 ESP-IDF Docker 镜像
docker pull espressif/idf

# 创建工作目录
mkdir -p ~/esp-mesh-project
cd ~/esp-mesh-project

# 启动容器
docker run --rm -v $PWD:/project -w /project -it espressif/idf bash

2. 创建项目

# 在容器内执行
idf.py create-project bluetooth-mesh-node
cd bluetooth-mesh-node

3. 配置 Mesh 参数

编辑 sdkconfig,启用蓝牙 Mesh 支持:

idf.py menuconfig

导航到:

Component config → Bluetooth → Bluedroid Enable → Enable Bluetooth Mesh

保存退出后,执行:

idf.py fullclean

代码实现

节点主程序

创建 main/main.c

#include 
#include "esp_log.h"
#include "nvs_flash.h"
#include "esp_bt.h"
#include "esp_ble_mesh_api.h"

static const char *TAG = "MESH_NODE";

// 设备 UUID(每个设备需要唯一)
static uint8_t dev_uuid[16] = {
    0xdd, 0xdd, 0xdd, 0xdd, 0xdd, 0xdd, 0xdd, 0xdd,
    0xdd, 0xdd, 0xdd, 0xdd, 0xdd, 0xdd, 0xdd, 0xdd
};

// Mesh 配置
static esp_ble_mesh_cfg_srv_t config_server = {
    .relay = ESP_BLE_MESH_RELAY_DISABLED,
    .beacon = ESP_BLE_MESH_BEACON_ENABLED,
};

static esp_ble_mesh_model_t models[] = {
    ESP_BLE_MESH_MODEL_CFG_SRV(&config_server),
    ESP_BLE_MESH_MODEL_NONE(),
};

static esp_ble_mesh_elem_t elements[] = {
    ESP_BLE_MESH_ELEMENT(0, models, models, models),
};

static esp_ble_mesh_comp_t composition = {
    .cid = 0x02E5,
    .elements = elements,
    .element_count = ARRAY_SIZE(elements),
};

static esp_ble_mesh_prov_t provision = {
    .uuid = dev_uuid,
    .uuid_size = sizeof(dev_uuid),
    .attention_duration = 3,
};

// Mesh 事件回调
static void mesh_callback(esp_ble_mesh_event_t *event) {
    switch (event->event) {
        case ESP_BLE_MESH_PROVISION_NODE_EVT:
            ESP_LOGI(TAG, "节点配网成功");
            break;
        case ESP_BLE_MESH_NODE_EVT:
            ESP_LOGI(TAG, "节点收到消息");
            break;
        default:
            break;
    }
}

void app_main(void) {
    esp_err_t ret;

    // 初始化 NVS
    ret = nvs_flash_init();
    if (ret == ESP_ERR_NVS_NO_FREE_PAGES) {
        ESP_ERROR_CHECK(nvs_flash_erase());
        ret = nvs_flash_init();
    }
    ESP_ERROR_CHECK(ret);

    // 初始化蓝牙
    ESP_ERROR_CHECK(esp_bt_controller_mem_release(ESP_BT_MODE_CLASSIC_BT));

    esp_bt_controller_config_t bt_cfg = BT_CONTROLLER_INIT_CONFIG_DEFAULT();
    ESP_ERROR_CHECK(esp_bt_controller_init(&bt_cfg));
    ESP_ERROR_CHECK(esp_bt_controller_enable(ESP_BT_MODE_BLE));

    // 初始化 Bluedroid
    ESP_ERROR_CHECK(esp_bluedroid_init());
    ESP_ERROR_CHECK(esp_bluedroid_enable());

    // 初始化 Mesh
    esp_ble_mesh_init(mesh_callback);

    // 开始配网
    ESP_ERROR_CHECK(esp_ble_mesh_provisioner_add_unprov_dev(
        dev_uuid, sizeof(dev_uuid),
        0, 0, NULL, 0
    ));

    ESP_LOGI(TAG, "蓝牙 Mesh 节点启动完成");
}

编译烧录

# 编译项目
idf.py build

# 烧录到 ESP32(替换 /dev/ttyUSB0 为你的设备端口)
idf.py -p /dev/ttyUSB0 flash monitor

组网配置

1. 配网(Provisioning)

使用手机 App(如 nRF Mesh)进行配网:

  1. $1

  2. $1

  3. $1

  4. $1

2. 配置发布/订阅

在 App 中配置:

设备:Living Room Light
├─ 订阅地址:0xC001(群组地址)
└─ 发布地址:0x0001(自身单播地址)

设备:Bedroom Sensor
├─ 订阅地址:0x0002(自身单播地址)
└─ 发布地址:0xC001(群组地址)

这样,传感器数据发布到 0xC001,所有订阅该地址的设备都能收到。

3. 创建场景

场景是一组预定义的操作:

场景名称:离家模式
触发条件:按下场景开关
执行动作:
  - 关闭客厅灯(地址 0x0001)
  - 关闭卧室灯(地址 0x0002)
  - 开启安防模式(地址 0x0003)

常见问题排查

问题 1:设备无法配网

症状:手机 App 扫描不到设备

排查步骤

  1. $1

  2. $1

  3. $1

  4. $1

问题 2:消息无法传递

症状:设备已配网,但控制指令无响应

排查步骤

  1. $1

  2. $1

  3. $1

  4. $1

问题 3:功耗过高

症状:电池供电设备续航短

优化方案

  1. $1

  2. $1

  3. $1

  4. $1

// 进入深度睡眠(10 秒后唤醒)
esp_deep_sleep(10 * 1000000ULL);

问题 4:覆盖范围不足

症状:远距离设备通信不稳定

解决方案

  1. $1

  2. $1

  3. $1

  4. $1

实战案例:智能灯光系统

让我们搭建一个完整的智能灯光系统:

系统架构

┌─────────────┐     ┌─────────────┐     ┌─────────────┐
│  场景开关   │────▶│  网关节点   │◀────│  手机 App   │
│  (0x0001)   │     │  (0x0002)   │     │             │
└─────────────┘     └──────┬──────┘     └─────────────┘

              ┌────────────┼────────────┐
              ▼            ▼            ▼
        ┌─────────┐  ┌─────────┐  ┌─────────┐
        │ 客厅灯  │  │ 卧室灯  │  │ 厨房灯  │
        │(0x0003) │  │(0x0004) │  │(0x0005) │
        └─────────┘  └─────────┘  └─────────┘

配置步骤

  1. $1

  2. $1

  3. $1

自动化联动

添加传感器实现自动化:

如果 光照传感器 < 100lux 且 人体传感器 = 检测到
则 开启对应区域灯光(延迟 30 秒自动关闭)

性能优化建议

1. 网络拓扑优化

  • 星型拓扑:适合小范围(<50 平米)

  • 网状拓扑:适合大范围,启用中继

  • 混合拓扑:关键路径用网状,边缘用星型

2. 消息优先级

// 高优先级消息(立即发送)
esp_ble_mesh_generic_server_publish(..., TTL_HIGH);

// 低优先级消息(可延迟)
esp_ble_mesh_generic_server_publish(..., TTL_LOW);

3. 安全加固

  • 启用应用密钥加密

  • 定期更新网络密钥

  • 限制配网时间窗口

  • 使用 OOB 认证(如二维码)

总结

蓝牙 Mesh 是搭建智能家居的务实选择。它成本低、覆盖广、功耗低,而且不需要额外网关。

关键要点:

  • 理解发布/订阅机制是组网的核心

  • 合理规划网络拓扑和地址分配

  • 低功耗场景启用 LPN 模式

  • 信号弱的区域增加中继节点

200 块的硬件投入,就能搭建一套覆盖全屋的智能控制系统。剩下的就是发挥创意,把你的想法变成现实。

希望这篇博客文章对您有所帮助!