国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

你不知道的前端藍牙應用實踐-心率帶

開發 前端
藍牙低能耗技術,實現設備間的連接與通信。顧名思義,能耗與成本都更低,與之相對應的則稱為經典藍牙。基于筆者的開發目的,本文簡單了解設備連接前(GAP)和連接后(GATT)所涉及的兩個協議。

本文為來自 字節教育-成人與創新前端團隊 成員的文章,已授權 ELab 發布。

一、背景

最近開啟了減肥計劃,購入了一條心率帶,期望在使用劃船機過程中監測心率情況。購入后的情況如下:

心率帶不直接顯示數值,需要連接APP或相關設備使用。

官方APP僅實時顯示心率數據,無法生成心率統計圖表。

圖片

圖片

通過咕咚APP連接心率帶,開啟運動后可以監測心率變化,但劃船機不在支持的運動范圍內。

自己簡單實現了一個劃船機節拍器的小程序。

圖片

圖片

圖片

于是萌生了在自己的節拍器小程序內監聽心率數據的想法,即Taro小程序中的藍牙應用實踐。

二、簡單了解藍牙

對于概念類的知識筆者畢竟不是專業的,感興趣的同學可以通過百科、搜索引擎等渠道進行了解。這里僅簡單介紹接下來會運用到的一些知識。

中心設備/外圍設備

客戶端(中心設備):在本次實踐中為筆者的手機。

服務端(外圍設備):在本次實踐中為心率帶、耳機等設備。

BLE(Bluetooth Low Energy)

藍牙低能耗技術,實現設備間的連接與通信。顧名思義,能耗與成本都更低,與之相對應的則稱為經典藍牙。基于筆者的開發目的,本文簡單了解設備連接前(GAP)和連接后(GATT)所涉及的兩個協議。

(了解更多:藍牙低能耗——百度百科[1]、一文讀懂藍牙技術從 1.0 到 5.0 的前世今生[2])

GAP(Generic Access Profile)

主要用來控制設備連接和廣播。通常是由外圍設備主動、間隔性地廣播設備信息,等待中心設備發現并建立連接。需要注意的是,這種連接方式是獨占的,在建立連接后,外圍設備將停止廣播。

另一方面,還存在僅向外廣播而不建立連接的iBeacon設備。

GATT(Generic Attribute Profile)

定義了兩個設備間的數據傳輸方式。GATT中有兩個關鍵概念:service(服務)與characteristic(特征)。

Service就是一個獨立的邏輯項,它包含一個或多個Characteristic。

Characteristic是GATT中最小的邏輯數據單元,其屬性包含properties,標識該特性是否能夠被read、write、notify、indicate。notify與indicate的區別在于,indicate需要有回復才能發送下一個數據包。

每個服務和特性都具有唯一的UUID標識,其中部分是由Bluetooth SIG官方定義的, Assigned Numbers | Bluetooth? Technology Website[3],如設備名、心率數據等常用屬性都是官方定義來統一規范。此外UUID也可以由硬件工程師來自定義實現。

(了解更多:BLE相關協議(GAP&GATT)[4])

三、API簡介

Taro中的藍牙API

Taro.openBluetoothAdapter(option) | Taro 文檔[5]

初始化藍牙模塊

Taro.openBluetoothAdapter({
success: function (res) {
console.log("藍牙環境已啟動:", res);
setInitStatus(true);
},
fail: function (err) {
if (err.errMsg.includes("fail already opened")) {
console.log("藍牙環境原先已啟動:", err);
setInitStatus(true);
} else {
console.log("藍牙環境啟動失?。?, err);
setInitStatus(false);
}
},
});

Taro.getBluetoothDevices(option) | Taro 文檔[6]

獲取在藍牙模塊生效期間所有已發現的藍牙設備。包括已經和本機處于連接狀態的設備。

discoverInterval = setInterval(() => {
Taro.getBluetoothDevices({
success: async function (res) {
const canConnectDevices = res.devices.filter(
(item) =>
// 信號強度大于-80
item.RSSI > -80 &&
// 含有設備名
!["未知設備", "MBeacon"].includes(item.name)
);
console.log("獲取藍牙設備列表成功:", canConnectDevices);

setDeviceList(() => canConnectDevices);
},
});
}, 2000);

也可以使用 Taro.onBluetoothDeviceFound(callback) | Taro 文檔[7]來發現設備。

Taro.createBLEConnection(option) | Taro 文檔[8]

連接低功耗藍牙設備。

若小程序在之前已有搜索過某個藍牙設備,并成功建立連接,可直接傳入之前搜索獲取的 deviceId 直接嘗試連接該設備,無需進行搜索操作。

Taro.createBLEConnection({
deviceId,
success: function (res) {
console.log("設備連接成功", res);
setConnectDeviceId(deviceId);

Taro.onBLEConnectionStateChange(function (res) {
// 該方法回調中可以用于處理連接意外斷開等異常情況
console.log(
`設備 ${res.deviceId}連接狀態發生變化: ${res.connected}`
);
if (!res.connected) {
setConnectDeviceId("");
}
});
},
});

Taro.getBLEDeviceServices(option) | Taro 文檔[9]

獲取藍牙設備所有服務(service)。

Taro.getBLEDeviceServices({
// 這里的 deviceId 需要已經通過 createBLEConnection 與對應設備建立鏈接
deviceId,
success: function (res) {
console.log('device services:', res.services)
}
})

Taro.getBLEDeviceCharacteristics(option) | Taro 文檔[10]

獲取藍牙設備某個服務中所有特征值(characteristic)。

Taro.getBLEDeviceCharacteristics({
// 這里的 deviceId 需要已經通過 createBLEConnection 與對應設備建立鏈接
deviceId,
// 這里的 serviceId 需要在 getBLEDeviceServices 接口中獲取
serviceId,
success: function (res) {
console.log('device getBLEDeviceCharacteristics:', res.characteristics)
}
})

Taro.readBLECharacteristicValue(option) | Taro 文檔[11]

讀取低功耗藍牙設備的特征值的二進制數據值。注意:必須設備的特征值支持 read 才可以成功調用。

接口讀取到的信息需要在onBLECharacteristicValueChange 方法注冊的回調中獲取。

Taro.notifyBLECharacteristicValueChange(option) | Taro 文檔[12]

啟用低功耗藍牙設備特征值變化時的 notify 功能,訂閱特征值。注意:必須設備的特征值支持 notify 或者 indicate 才可以成功調用。

另外,必須先啟用 notifyBLECharacteristicValueChange 才能監聽到設備 characteristicValueChange 事件

Taro.onBLECharacteristicValueChange(callback) | Taro 文檔[13]

監聽低功耗藍牙設備的特征值變化事件。必須先啟用 notifyBLECharacteristicValueChange 接口才能接收到設備推送的 notification。

WEB藍牙API

此處貼一些資料,感興趣可自行閱讀

Web Bluetooth API - Web APIs | MDN[14]

通過 JavaScript 與藍牙設備通信[15]

四、設備名稱(Device Name)詳解

首先getBLEDeviceServices獲取到服務列表:

圖片

查詢資料

圖片

可知0x1800是我們需要的服務。getBLEDeviceCharacteristics獲取其特征列表:

圖片

查詢資料

圖片

可知0x2A00是我們需要的特征。此時可以看到read屬性為true,我們通過onBLECharacteristicValueChange和readBLECharacteristicValue讀一下數據看看。

Taro.onBLECharacteristicValueChange(function (characteristic) {
const buffer = characteristic.value
const unit8Array = new Uint8Array(buffer);
console.log("unit8Array: ", unit8Array);

// 轉字符串
const encodedString = String.fromCodePoint.apply(null, unit8Array);
console.log('設備名:', encodedString)
});

Taro.readBLECharacteristicValue({
deviceId,
serviceId: DeviceNameService,
characteristicId: DeviceNameCharacteristics,
});

得到輸出,某米耳機:

圖片

圖片

大家應該已經發現,給到的特征值其實是ArrayBuffer格式。

(了解更多:談談JS二進制:File、Blob、FileReader、ArrayBuffer、Base64 - 掘金[16])

此時需要我們將其轉化為字符串。除了上面的方法外,還可以先轉16進制,再轉字符串:

// 轉16進制
const hexString = Array.prototype.map
.call(unit8Array, function (bit) {
return ("00" + bit.toString(16)).slice(-2);
})
.join("");
console.log("hexString: ", hexString);

// 16進制轉字符串
const hex2String = (hexString:string) => {
if (hexString.length % 2) return "";
var tmp = "";
for (let i = 0; i < hexString.length; i += 2) {
tmp += "%" + hexString.charAt(i) + hexString.charAt(i + 1);
}
return decodeURI(tmp);
}

五、心率測量(Heart Rate Measurement)詳解

同樣,首先我們拿到了所需的服務和特征如下。

export const HEART_RATE_SERVICE_UUID =
"0000180D-0000-1000-8000-00805F9B34FB";
export const HEART_RATE_CHARACTERISTIC_UUID =
"00002A37-0000-1000-8000-00805F9B34FB";

在設備連接后,通過onBLECharacteristicValueChange和notifyBLECharacteristicValueChange訂閱特征值變化。

export const blueToothGetHeartRate = (
deviceId: string,
onHeartRateChange: (newHeartRate: number) => void
) => {
Taro.onBLECharacteristicValueChange(function (characteristic) {
const heartRateValue = getHeartRateValue(characteristic.value);
onHeartRateChange(heartRateValue);
});
Taro.notifyBLECharacteristicValueChange({
state: true, // 啟用 notify 功能
deviceId,
serviceId: HEART_RATE_SERVICE_UUID,
characteristicId: HEART_RATE_CHARACTERISTIC_UUID,
});
};

此時我們已經能夠獲取到心率帶發送的心率數據如下。

圖片

但是此時如果按照上文解析設備名稱的方式,將其轉化為字符串,將得到一串亂碼。

圖片

所以需要根據協議文檔(https://www.bluetooth.com/specifications/specs/heart-rate-service-1-0/)來解讀這幾個數據。

標志字段:

  • 二進制:10110、十進制:22、十六進制:16
  • 其中位0為心率格式位,決定心率數據是uint8(位0 === 0)還是unit16(位0 === 1)

圖片

  • 位1、位2為傳感器接觸狀態位。11表示支持皮膚接觸檢測且接觸正常。
  • 位3為能量消耗狀態位,0表示能量消耗字段不存在.
  • 位4為RR-Interval狀態位,1表示存在一個或多個 RR-Interval 值。

心率數值:

  • 二進制:1100101、十進制:101、十六進制:65
  • 當心率小于255時,unit8足以。但如果需要支持更高的心率值(某些動物),則需要為unit16

圖片

  • 此時我們根據標志字段0,已經能夠獲取到需要的心率值:101。那么還剩下兩個字段代表什么意思呢?

RR-intreval 心率間隔:

  • 從前面的標志字段分析中,我們發現該設備不支持能量消耗狀態但支持RR-interval。且RR-interval可能是一位也可能是多位。那么我們該怎么讀取這個數字呢?
  • 首先我們顧名思義,心率間隔就是兩次心跳間的間隔時長,從上面的心率值推算:
  • 60*1000/101 ≈594ms
  • 接下來看這兩個數字。
  • 二進制:1010010、十進制:82、十六進制:52
  • 二進制:10、十進制:2、十六進制:02
  • 嘗試2*256+82 = 594,謎團解開了~

圖片

六、總結與疑惑

至此整個藍牙心率設備數據獲取的實現就完成了,可以在使用節拍器的同時監控自身的心率數據了。

圖片

整體來說整個開發過程還是比較簡單的,畢竟API文檔描述的非常清晰,主要時間耗費在解讀心率數據這個過程,后來知道應該從協議文檔出發解讀就好說了。

由于是在業余時間實現的上述能力,開發過程中存在不少疑惑沒來得及研究。這里先拋兩個問題出來,希望了解相關知識的同學能夠不吝賜教。

  • 在arraybuffer轉16進制過程中("00" + bit.toString(16)).slice(-2);?,為什么要先"00" +?,然后再.slice(-2)??直接bit.toString(16)是否可行?
  • 針對某米耳機,筆者暫時沒有在服務和特征中找到電量信息相關的數據,那么手機設備又是如何獲取到耳機電量的呢?

猜測電池服務和特性分別是0x180F Battery service 和 0x2A19 Battery Level ,但上圖耳機返回的service列表中沒找到該服務。

圖片

本文作者正在等待你的幫助。

參考資料

[1]藍牙低能耗——百度百科: https://baike.baidu.com/item/%E8%93%9D%E7%89%99%E4%BD%8E%E8%83%BD%E8%80%97

[2]一文讀懂藍牙技術從 1.0 到 5.0 的前世今生: https://zhuanlan.zhihu.com/p/37717509

[3]Assigned Numbers | Bluetooth? Technology Website: https://www.bluetooth.com/specifications/assigned-numbers/

[4]BLE相關協議(GAP&GATT): https://www.jianshu.com/p/62eb2f5407c9

[5]Taro.openBluetoothAdapter(option) | Taro 文檔: https://docs.taro.zone/docs/apis/device/bluetooth/openBluetoothAdapter

[6]Taro.getBluetoothDevices(option) | Taro 文檔: https://docs.taro.zone/docs/apis/device/bluetooth/getBluetoothDevices

[7]Taro.onBluetoothDeviceFound(callback) | Taro 文檔: https://docs.taro.zone/docs/apis/device/bluetooth/onBluetoothDeviceFound

[8]Taro.createBLEConnection(option) | Taro 文檔: https://docs.taro.zone/docs/apis/device/bluetooth-ble/createBLEConnection

[9]Taro.getBLEDeviceServices(option) | Taro 文檔: https://docs.taro.zone/docs/apis/device/bluetooth-ble/getBLEDeviceServices

[10]Taro.getBLEDeviceCharacteristics(option) | Taro 文檔: https://docs.taro.zone/docs/apis/device/bluetooth-ble/getBLEDeviceCharacteristics

[11]Taro.readBLECharacteristicValue(option) | Taro 文檔: https://docs.taro.zone/docs/apis/device/bluetooth-ble/readBLECharacteristicValue

[12]Taro.notifyBLECharacteristicValueChange(option) | Taro 文檔: https://docs.taro.zone/docs/apis/device/bluetooth-ble/notifyBLECharacteristicValueChange

[13]Taro.onBLECharacteristicValueChange(callback) | Taro 文檔: https://docs.taro.zone/docs/apis/device/bluetooth-ble/onBLECharacteristicValueChange

[14]Web Bluetooth API - Web APIs | MDN: https://developer.mozilla.org/en-US/docs/Web/API/Web_Bluetooth_API

[15]通過 JavaScript 與藍牙設備通信: https://web.dev/i18n/zh/bluetooth/

[16]談談JS二進制:File、Blob、FileReader、ArrayBuffer、Base64 - 掘金: https://juejin.cn/post/7148254347401363463#heading-9

責任編輯:武曉燕 來源: ELab團隊
相關推薦

2018-12-06 09:12:58

2020-06-12 09:20:33

前端Blob字符串

2020-07-28 08:26:34

WebSocket瀏覽器

2009-12-10 09:37:43

2022-10-13 11:48:37

Web共享機制操作系統

2021-02-01 23:23:39

FiddlerCharlesWeb

2011-09-15 17:10:41

2010-08-23 09:56:09

Java性能監控

2011-05-29 17:04:10

筆記本體驗

2022-11-04 08:19:18

gRPC框架項目

2020-09-15 08:35:57

TypeScript JavaScript類型

2021-10-17 13:10:56

函數TypeScript泛型

2020-08-11 11:20:49

Linux命令使用技巧

2015-06-19 13:54:49

2021-12-29 11:38:59

JS前端沙箱

2012-11-23 10:57:44

Shell

2021-12-22 09:08:39

JSON.stringJavaScript字符串

2015-10-23 08:51:18

應用暗知識發現

2025-03-17 00:45:00

JavaScriptAPI頁面

2012-06-26 15:49:05

點贊
收藏

51CTO技術棧公眾號

亚洲欧美清纯在线制服| 国产欧美精品一区二区三区-老狼| 久久久久久久久久久久久夜| 日韩中文字幕不卡| 亚洲欧美日产图| 国产秀色在线www免费观看| 国产91精品一区二区麻豆亚洲| 日本高清视频一区二区三区| 模特精品在线| 色一情一乱一伦一区二区三区 | 国产97在线|亚洲| 米奇777在线影院线| 日韩 欧美一区二区三区| 性欧美lx╳lx╳| 欧美中文字幕不卡| av电影高清在线观看| 亚洲一区999| 91成人噜噜噜在线播放| 国产精品久久久久久久久久ktv | 国产91在线播放精品91| 免费精品视频| 欧美激情精品久久久久久免费印度| 国产美女精品视频免费播放软件 | 秋霞午夜av一区二区三区| 视频一区不卡| 亚洲欧洲日韩一区二区三区| 久久色中文字幕| 一区二区传媒有限公司| 日韩毛片在线免费看| 久久中文字幕电影| 欧洲一区av| 日韩在线视频观看| 亚洲国产精品一区制服丝袜| 日韩免费视频播放| 亚洲第一男人天堂| 国产高清在线a视频大全| 亚洲欧美在线免费观看| 国产中文在线视频| 成年人视频免费在线播放| 日韩成人精品| 中日韩男男gay无套| 欧美国产精品中文字幕| 91精品综合久久久久久| 成人精品福利视频| 尤蜜粉嫩av国产一区二区三区| 新版的欧美在线视频| 久久不射中文字幕| 日韩av影视综合网| 免费在线观看污污视频| 1024国产在线| 高清电影在线免费观看| a亚洲天堂av| 久久亚洲精品成人| 99精品免费在线观看| 99热这里有精品| 国产精品美女久久久久久2018| 日本老师69xxx| 蝌蚪视频在线播放| 欧美一级一区| 538国产精品视频一区二区| 一个人免费观看视频www在线播放| 久久九九99| 久久久女女女女999久久| 欧美18hd| 欧美性xxxxx极品| 欧美在线观看www| 欧美成人tv| 欧美另类极品videosbest最新版本| 在线视频国产三级| 不卡的看片网站| 国产高清精品一区二区| 北条麻妃在线一区| 亚洲精品一线二线三线| 亚洲制服欧美另类| 国产裸体舞一区二区三区 | 亚洲无线码在线一区观看| 成人看片爽爽爽| 成人在线国产精品| 精品一区二区在线看| 99热免费在线观看| 蜜桃免费网站一区二区三区| 麻豆视频免费在线观看| 日韩免费一区二区| 99视频一区| 波多野结衣av在线播放| 精品香蕉一区二区三区| 成人在线中文| 日本精品视频一区| 欧美理论电影大全| 国产又爽又黄ai换脸| 舔着乳尖日韩一区| 欧美天堂在线| 三级三级久久三级久久18| 香蕉久久一区二区不卡无毒影院| 日韩欧美精品电影| 欧美久久综合性欧美| 亚洲大片精品永久免费| 国产成人免费视频网站视频社区 | 亚洲福利一区二区三区| 波多野结衣在线高清| 亚洲一区二区3| 成人午夜亚洲| 亚洲看片网站| 天天综合天天综合色| 成人福利一区| cao在线观看| 亚洲国产精品va在线看黑人 | 老司机午夜精品视频在线观看| 成人网免费看| 久久99久久99精品免观看粉嫩 | 色婷婷综合久色| 香蕉久久精品日日躁夜夜躁| 大肉大捧一进一出好爽视频| 亚洲美女福利视频网站| 日韩中文字幕av电影| 午夜在线免费观看视频| 91精品免费| 日韩欧美中文第一页| 三区四区不卡| 三级国产三级在线| 国产精品久久久久久婷婷天堂| 国产精品免费av| 日韩一区二区三区精品 | 国产**成人网毛片九色| 俄罗斯一级**毛片在线播放| 欧美日韩喷水| 9191精品国产综合久久久久久| 综合天天久久| 欧美色18zzzzxxxxx| 亚洲一区中文字幕在线观看| 性做久久久久久免费观看欧美| 久久91精品| 欧美日韩免费看| 91精品国产丝袜白色高跟鞋| 亚洲国产精品久久不卡毛片| www.99riav| 亚洲一级黄色片| 成人午夜视频在线观看| 欧美一级大黄| 日本一区午夜艳熟免费| www.亚洲免费视频| 国产精品水嫩水嫩| 欧美男同视频网| 青青草免费观看免费视频在线| 97免费资源站| 欧美一区二区三区性视频| 久久99久国产精品黄毛片色诱| 自拍在线观看| 欧美a v在线播放| 午夜精品99久久免费| 亚洲人被黑人高潮完整版| 国产精品成人a在线观看| 国产中文字幕在线看| 亚洲二区自拍| 久久久www成人免费精品张筱雨| 国产日本欧美一区二区| 日韩国产一区| 免费的黄网站在线观看| 国产片侵犯亲女视频播放| 久久久久久久久国产| 欧美日韩免费在线观看| 青草国产精品久久久久久| 国产精品久久久久77777丨| 成人毛片免费在线观看| 国产精品美女xx| 亚洲图片在区色| 一区二区三区在线视频播放| 国产亚洲高清视频| 久久亚洲人体| 羞羞视频在线免费看| 色一情一乱一伦一区二区三区丨| 久久精品亚洲国产| 欧美色视频日本版| 国内久久婷婷综合| 特黄特色欧美大片| 色视频在线免费观看| 分分操这里只有精品| 国产成人在线一区| 欧美va日韩va| 国产精品日韩成人| 亚洲黄页一区| 日韩在线观看中文字幕| a√资源在线| 18禁男女爽爽爽午夜网站免费 | 成人国产精品一区二区免费麻豆 | 国产精品久久久久免费a∨| 日韩视频免费观看高清完整版在线观看 | 欧美一级视频免费看| 免费av一区二区三区| 国产日韩在线视频| 性欧美办公室18xxxxhd| 狠狠网亚洲精品| 国产日韩av网站| 国产精品视频自在线| 精品国产麻豆免费人成网站| 中文字幕日韩一区| 蜜桃一区二区三区四区| 一本色道久久综合亚洲精品酒店 | 亚洲精品福利在线| 国产精品日韩av|