Rokid AI 眼镜远程协作应用"一线互联"开发实践:设备发现与 BLE 扫描

简介: 工业现场蓝牙扫描远非简单“搜到即连”:需两级过滤(系统低延迟扫描+应用层UUID/名称匹配)、RSSI智能排序、MAC地址临时命名,并通过CXR私有协议建立专用连接,确保秒级稳定接入。

上周在工厂部署一线互联的时候,一个工程师问我:"不就是蓝牙扫描嘛,搜到了点连接不就行了?"

行是行,但在工业现场,"搜到了"这三个字没那么简单。
iShot_2026-06-01_11.10.11.png

你周围不止一副蓝牙设备

掏出手机扫一下蓝牙,列表里几十个设备。耳机、手环、车间里的蓝牙网关、隔壁工位的手机——Rokid AI 眼镜就混在里面。问题是,眼镜的 BLE 广播名有时候是空的,有时候显示为乱码,你不能只靠名字来认。

我们把扫描过滤做成了两级:

第一级是 Android 系统层的 BLE 扫描配置。用的是低延迟模式,牺牲一点功耗换扫得更快——工厂讲究戴上就干活,不能让人等着蓝牙慢慢扫:

val settings = ScanSettings.Builder()
    .setScanMode(ScanSettings.SCAN_MODE_LOW_LATENCY)
    .build()

scanner.startScan(null, settings, scanCallback)

第二级是应用层的过滤。每个扫描结果进来,我们会检查设备名和 Service UUID 是否匹配 Rokid 眼镜的特征。如果两个都不匹配,直接丢掉:

override fun onScanResult(callbackType: Int, result: ScanResult) {
    val device = result.device ?: return
    val serviceUuids = result.scanRecord?.serviceUuids
        ?.map { it.uuid.toString() } ?: emptyList()
    val name = result.scanRecord?.deviceName ?: runCatching { device.name }.getOrNull()

    if (!RokidGlassesBtScanFilter.matches(
            name = name,
            serviceUuids = serviceUuids,
            targetServiceUuid = config.serviceUuid,
            allowNameFallback = config.allowNameFallback
        )
    ) return

    // 只保留匹配的设备
    val displayName = name ?: "Glass_${device.address.takeLast(5).replace(":", "")}"
    upsertDevice(RokidGlassesBtDevice(
        name = displayName,
        address = device.address,
        type = device.type,
        rssi = result.rssi,
        serviceUuids = serviceUuids
    ))
}

这里有个小细节:如果广播里拿不到设备名,我们会用 MAC 地址后五位生成一个"Glass_XXXX"的临时名称。在车间里可能同时有好几副眼镜,这个名字至少能让用户区分。

还有一个实操经验值得提:扫描到的设备按 RSSI 降序排列。信号强的排上面,大概率就是离你最近的那副。上层的代码通过 StateFlow 拿到设备列表后直接绑定到 UI,每次新设备进来自动排序:

private fun upsertDevice(device: RokidGlassesBtDevice) {
    val next = _devices.value
        .filterNot { it.address.equals(device.address, ignoreCase = true) }
        .plus(device)
        .sortedByDescending { it.rssi }
    _devices.value = next
}

为什么不用传统蓝牙配对

这是另一个经常被问到的问题。普通蓝牙耳机、音箱用的是经典蓝牙 SPP/A2DP profile,配对完系统就记住了。但 Rokid 眼镜的视频流和控制指令走的是 CXR 私有协议通道,底层虽然也是蓝牙传输,但连接建立逻辑完全不一样。

打个比方:普通蓝牙设备像是标准快递,手机操作系统认识这个快递单,帮你签收了。但 CXR 通道像是专用物流,快递单和签收流程都是定制的,你得通过 CXR SDK 去初始化、协商 endpoint、建立 socket 连接。

这也是为什么一线互联在封装连接层的时候,要把 CXR SDK 的调用全部隔离在一个内部接口后面:

internal object AndroidRokidCxrClient : RokidCxrClient {
    override fun initBluetooth(
        context: Context,
        device: BluetoothDevice,
        callback: BluetoothStatusCallback
    ) {
        CxrApi.getInstance().initBluetooth(context, device, callback)
    }

    override fun connectBluetooth(
        context: Context,
        socketUuid: String,
        macAddress: String,
        bluetoothClientName: String,
        callback: BluetoothStatusCallback,
        authBlob: ByteArray,
        clientSecret: String
    ) {
        CxrApi.getInstance().connectBluetooth(
            context, socketUuid, macAddress, bluetoothClientName,
            callback, authBlob, clientSecret
        )
    }

    override fun deinitBluetooth() {
        CxrApi.getInstance().deinitBluetooth()
    }

    override fun isBluetoothConnected(): Boolean {
        return CxrApi.getInstance().isBluetoothConnected
    }
}

对上层业务来说,它只知道调用 manager.startScan()manager.connect(device),完全不用关心底层是 CXR 还是别的什么协议。如果未来 Rokid SDK 升级了 API 签名,改动的范围也只在这个文件里。

这篇文章其实就想说一件事:蓝牙扫描不是搜一下就行,设备过滤、排序、连接建立,每个环节的细节决定了工业现场的体验。把 BLE 扫描的五秒变成一秒,把"连不上"变成"三秒重连",就是这些细节堆出来的。



相关仓库:github.com/jlink-ai/rokid-glasses-bt · Apache 2.0

下一篇聊聊连接状态机和失败模型——也就是那个"SOCKET_CONNECT_FAILED"到底是怎么来的,以及为什么每一种错误都应该对应一个明确的操作指引。

相关文章
|
17天前
|
人工智能 数据可视化 5G
一线互联 × Rokid AI眼镜:为什么它是工业4.0时代一线人员的标准装备?
Gartner预测:2027年30%工业企业将为一线员工标配AI眼镜(2023年仅5%)。Rokid灵伴眼镜+一线互联jLink,以轻量化硬件、5G低延时网络与任务驱动型协作系统,实现远程指导、过程留痕、数据归档闭环,助力企业降本增效、沉淀数字资产。(239字)
103 1
|
3月前
|
人工智能 数据挖掘 Linux
阿里云/本地部署OpenClaw 配置 AI Agent团队实战:从1扩至8再精简为4的完整复盘+避坑指南
在AI Agent全面落地的2026年,越来越多用户尝试用OpenClaw(Clawdbot)搭建专属AI团队,替代人工完成内容创作、运营分发、技术维护、数据分析等工作。但很多人在搭建过程中陷入盲目扩张误区:从单个智能体起步,不断新增角色,最终导致管理成本飙升、上下文断裂、产出质量下滑、系统资源浪费。
720 2
|
数据安全/隐私保护
CobaltStrike 流量隐藏
CobaltStrike 流量隐藏
346 0
|
1月前
|
安全 关系型数据库 API
跨平台办公数据集成:钉钉与飞书表格自动化的工程实践
本文提出轻量级跨平台数据集成方案,通过节点化抽象钉钉表格与飞书多维表操作,借助流程编排引擎实现双向同步;支持定时/事件触发,覆盖财务汇总、周报生成等场景,并可在阿里云上集成RDS、FC、SLS等服务提升稳定性与可观测性。(239字)
|
9月前
|
存储 供应链 JavaScript
ERP系统生产管理全流程解析
ERP系统为企业提供集成化生产管理解决方案,涵盖从销售预测、主生产计划、物料需求计划到生产订单下达、过程控制及成品入库的全流程。通过规范生产流程,提升计划准确性与执行效率,实现资源优化配置。实施时需注重前期规划、员工培训、数据管理及持续优化,确保系统有效支撑企业生产运营。
608 3
|
17天前
|
存储 弹性计算 人工智能
购买阿里云服务器省钱教程:分享给大家(学生、个人和企业用户都有省钱方法)
阿里云省钱购机指南:学生可领300元无门槛代金券,个人/企业用户享折扣券+特惠活动。推荐38元/年轻量服务器、99元/年ECS及199元/年通用型实例,叠加优惠更划算。活动中心与权益中心实时更新补贴,认证学生还可0元购机。阿里云官方活动:https://t.aliyun.com/U/OTnSAH
151 0
|
10月前
|
人工智能 算法 C++
浅谈 KMP
KMP算法是一种高效的字符串匹配算法,由Knuth、Morris和Pratt提出。它通过预处理模式串构建next数组,利用匹配失败时的信息减少重复比较,从而提升匹配效率。其时间复杂度为O(m+n),适用于大规模文本匹配场景。
621 0
|
安全 Shell Linux
openEuler OECA认证课程习题答案
本文整理了OECA认证课程的每章节课后习题答案。
916 2
openEuler OECA认证课程习题答案
|
弹性计算 大数据 双11
阿里云ECS“99套餐”再升级!双11一站式满足全年算力需求
弹性计算双11爆款清单来袭,感恩回馈好礼,优惠惊喜不断!
|
存储 固态存储 NoSQL
neo4j安装调研简介-代做找我
neo4j安装调研简介-代做找我
361 0

热门文章

最新文章