博客專欄

EEPW首頁(yè) > 博客 > 藍(lán)牙BLE MTU規(guī)則與約定

藍(lán)牙BLE MTU規(guī)則與約定

發(fā)布人:電子禪石 時(shí)間:2021-05-15 來(lái)源:工程師 發(fā)布文章

1. 問(wèn)題引言:

想在gatt client上(一般是手機(jī)上)傳輸長(zhǎng)一點(diǎn)的數(shù)據(jù)給gatt server(一般是一個(gè)Bluetooth smart設(shè)備,即只有BLE功能的設(shè)備),但通過(guò)

[java]

writeCharacteristic(BluetoothGattCharacteristic) 

來(lái)寫的時(shí)候發(fā)現(xiàn)最多只能寫入20個(gè)byte的數(shù)據(jù)。

這篇文章會(huì)回答下面幾個(gè)問(wèn)題:

1)為什么會(huì)是20?

2)如何突破20?

3)如何更優(yōu)雅的來(lái)實(shí)現(xiàn)?

2. 為什么為限制成20個(gè)字節(jié)?

core spec里面定義了ATT的默認(rèn)MTU23個(gè)bytes, 除去ATTopcode一個(gè)字節(jié)以及ATThandle 2個(gè)字節(jié)之后,剩下的20個(gè)字節(jié)便是留給GATT的了。

考慮到有些Bluetooth smart設(shè)備功能弱小,不敢太奢侈的使用內(nèi)存空間,因此core spec規(guī)定每一個(gè)設(shè)備都必須支持MTU23

在兩個(gè)設(shè)備連接初期,大家都像新交的朋友一樣,不知對(duì)方底細(xì),因此嚴(yán)格的按照套路來(lái)走,即最多一次發(fā)20個(gè)字節(jié),是最保險(xiǎn)的。

由于ATT的最大長(zhǎng)度為512byte

因此一般認(rèn)為MTU的最大長(zhǎng)度為512個(gè)byte就夠了,再大也沒(méi)什么意義,你不可能發(fā)一個(gè)超過(guò)512ATT的數(shù)據(jù)。

所以ATTMTU的最大長(zhǎng)度可視為512個(gè)bytes。

3. 如何突破20?

很簡(jiǎn)單嘛,改變傳輸?shù)?span style="overflow-wrap: break-word;">ATT的MTU就行了,大家經(jīng)過(guò)友好的協(xié)商,得到雙方都想要的結(jié)果,是最好的。在Android上(API 21),改變ATT MTU的接口為:

public boolean requestMtu (int mtu)    

Added in API level 21 

Request an MTU size used for a given connection. 

When performing a write request operation (write without response), the data sent is truncated to the MTU size. This function may be used to request a larger MTU size to be able to send more data at once. 

 A onMtuChanged(BluetoothGatt, int, int) callback will indicate whether this operation was successful. 

  Requires BLUETOOTH permission. 

 

Returns 

true, if the new MTU value has been requested successfully 

大聲的說(shuō)出來(lái)你想要一下子傳多少,調(diào)用上面的接口就可以了,然后在下面的函數(shù)中看最終結(jié)果(當(dāng)然了,如果你的peripheral申請(qǐng)改變MTU并且成功的話,那這個(gè)回調(diào)也會(huì)被調(diào)用):

@Override 

public void onMtuChanged(BluetoothGatt gatt, int mtu, int status) { 

    super.onMtuChanged(gatt, mtu, status); 

 

    if (status == BluetoothGatt.GATT_SUCCESS) { 

        this.supportedMTU = mtu;//local var to record MTU size 

    } 

之后你就可以快樂(lè)的發(fā)送supported  MTU-3的長(zhǎng)度的數(shù)據(jù)了。

                                                                                                

4. 如何優(yōu)雅的來(lái)實(shí)現(xiàn)?

萬(wàn)一對(duì)方設(shè)備不同意你的請(qǐng)求怎么辦?

對(duì)于app來(lái)說(shuō),一般是知道自己要最****送多少數(shù)據(jù)的,例如一次要發(fā)100個(gè)bytes,那么就首先試試申請(qǐng)一下103,失敗的話,則申請(qǐng)一下53,即二分法,剩下的只能自己分段拆著發(fā)了。

一般來(lái)講,app的開(kāi)發(fā)者和對(duì)端設(shè)備的開(kāi)發(fā)者都是同一伙兒人,這是好事,他們可以根據(jù)自己設(shè)備的硬件情況,來(lái)商量MTU應(yīng)該是多少。

藍(lán)牙BLE MTU規(guī)則與約定-zhenhuaqin-ChinaUnix博客

*博客內(nèi)容為網(wǎng)友個(gè)人發(fā)布,僅代表博主個(gè)人觀點(diǎn),如有侵權(quán)請(qǐng)聯(lián)系工作人員刪除。



關(guān)鍵詞: bluetooth

相關(guān)推薦

技術(shù)專區(qū)

關(guān)閉