Android如何實現(xiàn)藍牙配對連接功能
本文適用的范圍
Android藍牙部分是很復(fù)雜的,也涉及很多名詞和功能。本文介紹的配對連接方法適用于一般的藍牙耳機、音響等,并不是連接藍牙 BLE 或者想用藍牙來進行 Socket 通信的。
先來介紹幾種名稱:
Profile:
- Bluetooth 的一個很重要特性,就是所有的 Bluetooth 產(chǎn)品都無須實現(xiàn)全部的 Bluetooth 規(guī)范。為了更容易的保持 Bluetooth 設(shè)備之間的兼容,Bluetooth 規(guī)范中定義了 Profile。Profile 定義了設(shè)備如何實現(xiàn)一種連接或者應(yīng)用,你可以把 Profile 理解為連接層或者應(yīng)用層協(xié)。我們標題中的說的連接其實就是去連接各種 Profile。下面介紹的幾種都是Android 實現(xiàn)了的 Profile。
A2dp:
- 表示藍牙立體聲,和藍牙耳機聽歌有關(guān)那些,另還有個Avrcp音頻/視頻遠程控制配置文件,是用來聽歌時暫停,上下歌曲選擇的。
Handset、Handfree:
- 和電話相關(guān),藍牙接聽、掛斷電話。
其他:
- btservice關(guān)于藍牙基本操作的目錄,一切由此開始; hdp藍牙關(guān)于醫(yī)療方面的應(yīng)用;hid:人機交互接口,藍牙鼠標鍵盤什么的就是這個了 ;pbap:電話號碼簿訪問協(xié)議(Phonebook Access Profile) ...
準備
在 AndroidManifest.xml 添加所需的權(quán)限
<uses-permission android:name="android.permission.BLUETOOTH" /> <uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />
打開藍牙
mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter(); if (!mBluetoothAdapter.isEnabled()) { Intent enableIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE); startActivityForResult(enableIntent, REQUEST_ENABLE_BT); }
注冊廣播
由于藍牙的搜索、配對和連接狀態(tài)的改變都是系統(tǒng)通過廣播的方式發(fā)出來的,所以需要注冊這些廣播來獲取狀態(tài)的改變。
IntentFilter intentFilter = new IntentFilter(); intentFilter.addAction(BluetoothDevice.ACTION_FOUND); intentFilter.addAction(BluetoothAdapter.ACTION_DISCOVERY_FINISHED); intentFilter.addAction(BluetoothDevice.ACTION_BOND_STATE_CHANGED); intentFilter.addAction(BluetoothA2dp.ACTION_CONNECTION_STATE_CHANGED); intentFilter.addAction(BluetoothHeadset.ACTION_CONNECTION_STATE_CHANGED); registerReceiver(mReceiver, intentFilter);
搜索
獲取已配對的設(shè)備。對于之前已經(jīng)配對成功的設(shè)備,系統(tǒng)會把它的信息存儲在本地。再去調(diào)用搜索的時候,系統(tǒng)是不會重新再次發(fā)現(xiàn)這個設(shè)備的。
Set<BluetoothDevice> pairedDevices = mBluetoothAdapter.getBondedDevices();
搜索設(shè)備
mBluetoothAdapter.startDiscovery();
系統(tǒng)發(fā)現(xiàn)新的藍牙設(shè)備了之后,會通過廣播把這個設(shè)備的信息發(fā)送出來。所以我們要通過截獲 Action 為BluetoothDevice.ACTION_FOUND的 Intent,并得到設(shè)備信息。
BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
配對
重點來來了,做了一系列準備工作,拿到了BluetoothDevice下面就要開始配對連接了。但是坑的地方也在這里,首先藍牙設(shè)備必須要先配對成功了再去連接各個不同的 Profile,如果直接去連接有的機型確實也可以連上,但是大部分的都沒反應(yīng)。然后就是 Android 4.4 API 19 以上才開放配對接口,對于之前的系統(tǒng)我們只能通過反射的方式去獲取接口。
配對
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { //Android 4.4 API 19 以上才開放Bond接口 device.createBond(); } else { //API 19 以下用反射調(diào)用Bond接口 try { device.getClass().getMethod("connect").invoke(device); } catch (Exception e) { e.printStackTrace(); } }
配對成功會發(fā)送廣播BluetoothDevice.ACTION_BOND_STATE_CHANGED
//設(shè)備綁定狀態(tài)改變 BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE); int bondState = intent.getIntExtra(BluetoothDevice.EXTRA_BOND_STATE, BluetoothDevice.ERROR); //收到綁定成功的通知后自動連接 if (item != null && bondState == BluetoothDevice.BOND_BONDED) { connectDevice(item); }
配對的幾種狀態(tài)
public static final int BOND_NONE = 10; public static final int BOND_BONDING = 11; public static final int BOND_BONDED = 12;
連接
配對(綁定)和連接是兩個不同的過程,配對是指兩個設(shè)備發(fā)現(xiàn)了對方的存在,可以獲取到對方的名稱、地址等信息,有能力建立起連接。連接是指兩個設(shè)備共享了一個 RFCOMM 通道,有能力進行數(shù)據(jù)互傳。確認綁定上了之后,才能開始連接,連接其實就是連接這個藍牙設(shè)備支持的 Profile 。
可以觀察一下設(shè)置里面藍牙連接的過程過程,就是先開始配對,配對成功后才開始連接所有支持的 Profile。這一步也是比較坑的地方,網(wǎng)上都沒有詳細的對這一塊說明。我也是在 Setting 的源碼里面翻了半天才找到這塊的邏輯。但是系統(tǒng)應(yīng)用可以直接調(diào)用連接的方法,卻不外開放...
首先我們要提前獲取 Profile,這里拿A2dp來舉例,其他的原理是一樣的。
mBluetoothAdapter.getProfileProxy(this, new BluetoothProfile.ServiceListener() { @Override public void onServiceConnected(int profile, BluetoothProfile proxy) { if (mA2dpService == null) { mA2dpService = (BluetoothA2dp) proxy; } } @Override public void onServiceDisconnected(int profile) { } }, BluetoothProfile.A2DP);
當(dāng)我們收到配對成功的廣播或者確定設(shè)備已經(jīng)配對成功后,我們就要調(diào)用 Profile 的connect方法來連接。但是這個方法被 Google 給@hide了。像上面一樣用反射...
try { mA2dpService.getClass().getMethod("connect", BluetoothDevice.class) .invoke(mA2dpService, item.getDevice()); } catch (Exception e) { e.printStackTrace(); }
連接成功系統(tǒng)會發(fā)送廣播BluetoothA2dp.ACTION_CONNECTION_STATE_CHANGED
BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE); int profileState = intent.getIntExtra(BluetoothProfile.EXTRA_STATE, BluetoothProfile.STATE_DISCONNECTED);
連接的幾種狀態(tài)
/** The profile is in disconnected state */ public static final int STATE_DISCONNECTED = 0; /** The profile is in connecting state */ public static final int STATE_CONNECTING = 1; /** The profile is in connected state */ public static final int STATE_CONNECTED = 2; /** The profile is in disconnecting state */ public static final int STATE_DISCONNECTING = 3;
坑坑坑
哈哈,你以為連接上了就完事了嗎?!這里面還有幾個坑容我給你說說。
不要忘記關(guān)閉 Profile。我們?yōu)榱诉B接不是獲取了 Profile 嗎,在連接完成之后一定要關(guān)閉掉他,一直不關(guān)閉的話會報錯。
mBluetoothAdapter.closeProfileProxy(BluetoothProfile.A2DP, mA2dpService); mA2dpService = null;
藍牙連接成功了之后系統(tǒng)會通知手機狀態(tài)發(fā)生了改變,就像切換了橫豎屏一樣,需要重新調(diào)用這個 Activity 的生命周期。當(dāng)時我發(fā)現(xiàn)只要我一連接成功,我的界面就閃一下回到初始狀態(tài)。就納悶了半天,然后我一直以為是用反射連接導(dǎo)致程序異常重啟了...幾經(jīng)摸索我發(fā)現(xiàn)是沒有設(shè)置android:configChanges的緣故。
android:configChanges="keyboard|keyboardHidden|navigation"
其實前面兩個屬性我也早想到了,唯獨最后一個,在官方文檔里面寫的This should never normally happen.我天真的相信了,一直沒試它。最后實在沒辦法了把所有的屬性都寫上去,然后一個個減,最終發(fā)現(xiàn)了這三少一個都不行。
值 | 說明 |
---|---|
"keyboard" | 鍵盤類型發(fā)生了變化 — 例如,用戶插入了一個外置鍵盤。 |
"keyboardHidden" | 鍵盤無障礙功能發(fā)生了變化 — 例如,用戶顯示了硬件鍵盤。 |
"navigation" | 導(dǎo)航類型(軌跡球/方向鍵)發(fā)生了變化。(這種情況通常永遠不會發(fā)生。) |
以上就是Android如何實現(xiàn)藍牙配對連接功能的詳細內(nèi)容,更多關(guān)于Android 藍牙配對連接功能的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
詳解Android中Activity的啟動模式及應(yīng)用場景
今天給大家介紹下安卓開發(fā)中不得不涉及的Activity啟動模式及應(yīng)用場景,Activity一共有四種啟動模式,分別是Standard模式、SingleTop模式、SingleTask模式以及SingleInstance模式,,需要的朋友可以參考下2023-09-09Android中URLEncoder空格被轉(zhuǎn)碼為"+"號的處理辦法
當(dāng)上傳文件的文件名中間有空格,用URLEncoder.encode方法會把空格變成加號(+)在前臺頁面顯示的時候會多出加號,下面這篇文章主要給大家介紹了關(guān)于Android中URLEncoder空格被轉(zhuǎn)碼為"+"號的處理辦法,需要的朋友可以參考下2023-01-01Android獲得當(dāng)前正在顯示的activity類名的方法
這篇文章主要介紹了Android獲得當(dāng)前正在顯示的activity類名的方法,分析了權(quán)限的修改與Java代碼的實現(xiàn)技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-01-01在Android中查看當(dāng)前Activity是否銷毀的操作
這篇文章主要介紹了在Android中查看當(dāng)前Activity是否銷毀的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-03-03