Android如何實現(xiàn)藍牙配對連接功能
Android藍牙部分是很復雜的,也涉及很多名詞和功能。本文介紹的配對連接方法適用于一般的藍牙耳機、音響等,并不是連接藍牙 BLE 或者想用藍牙來進行 Socket 通信的。
先來介紹幾種名稱:
Profile:
Bluetooth 的一個很重要特性,就是所有的 Bluetooth 產(chǎn)品都無須實現(xiàn)全部的 Bluetooth 規(guī)范。為了更容易的保持 Bluetooth 設備之間的兼容,Bluetooth 規(guī)范中定義了 Profile。Profile 定義了設備如何實現(xiàn)一種連接或者應用,你可以把 Profile 理解為連接層或者應用層協(xié)。我們標題中的說的連接其實就是去連接各種 Profile。下面介紹的幾種都是Android 實現(xiàn)了的 Profile。A2dp:
表示藍牙立體聲,和藍牙耳機聽歌有關那些,另還有個Avrcp音頻/視頻遠程控制配置文件,是用來聽歌時暫停,上下歌曲選擇的。Handset、Handfree:
和電話相關,藍牙接聽、掛斷電話。其他:
btservice關于藍牙基本操作的目錄,一切由此開始; hdp藍牙關于醫(yī)療方面的應用;hid:人機交互接口,藍牙鼠標鍵盤什么的就是這個了 ;pbap:電話號碼簿訪問協(xié)議(Phonebook Access Profile) ...準備在 AndroidManifest.xml 添加所需的權限
<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);搜索
獲取已配對的設備。對于之前已經(jīng)配對成功的設備,系統(tǒng)會把它的信息存儲在本地。再去調(diào)用搜索的時候,系統(tǒng)是不會重新再次發(fā)現(xiàn)這個設備的。
Set<BluetoothDevice> pairedDevices = mBluetoothAdapter.getBondedDevices();
搜索設備
mBluetoothAdapter.startDiscovery();
系統(tǒng)發(fā)現(xiàn)新的藍牙設備了之后,會通過廣播把這個設備的信息發(fā)送出來。所以我們要通過截獲 Action 為BluetoothDevice.ACTION_FOUND的 Intent,并得到設備信息。
BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);配對
重點來來了,做了一系列準備工作,拿到了BluetoothDevice下面就要開始配對連接了。但是坑的地方也在這里,首先藍牙設備必須要先配對成功了再去連接各個不同的 Profile,如果直接去連接有的機型確實也可以連上,但是大部分的都沒反應。然后就是 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
//設備綁定狀態(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;連接
配對(綁定)和連接是兩個不同的過程,配對是指兩個設備發(fā)現(xiàn)了對方的存在,可以獲取到對方的名稱、地址等信息,有能力建立起連接。連接是指兩個設備共享了一個 RFCOMM 通道,有能力進行數(shù)據(jù)互傳。確認綁定上了之后,才能開始連接,連接其實就是連接這個藍牙設備支持的 Profile 。
可以觀察一下設置里面藍牙連接的過程過程,就是先開始配對,配對成功后才開始連接所有支持的 Profile。這一步也是比較坑的地方,網(wǎng)上都沒有詳細的對這一塊說明。我也是在 Setting 的源碼里面翻了半天才找到這塊的邏輯。但是系統(tǒ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);
當我們收到配對成功的廣播或者確定設備已經(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;坑坑坑
哈哈,你以為連接上了就完事了嗎?!這里面還有幾個坑容我給你說說。
不要忘記關閉 Profile。我們?yōu)榱诉B接不是獲取了 Profile 嗎,在連接完成之后一定要關閉掉他,一直不關閉的話會報錯。
mBluetoothAdapter.closeProfileProxy(BluetoothProfile.A2DP, mA2dpService);mA2dpService = null;
藍牙連接成功了之后系統(tǒng)會通知手機狀態(tài)發(fā)生了改變,就像切換了橫豎屏一樣,需要重新調(diào)用這個 Activity 的生命周期。當時我發(fā)現(xiàn)只要我一連接成功,我的界面就閃一下回到初始狀態(tài)。就納悶了半天,然后我一直以為是用反射連接導致程序異常重啟了...幾經(jīng)摸索我發(fā)現(xiàn)是沒有設置android:configChanges的緣故。
android:configChanges='keyboard|keyboardHidden|navigation'
其實前面兩個屬性我也早想到了,唯獨最后一個,在官方文檔里面寫的This should never normally happen.我天真的相信了,一直沒試它。最后實在沒辦法了把所有的屬性都寫上去,然后一個個減,最終發(fā)現(xiàn)了這三少一個都不行。
值 說明 'keyboard' 鍵盤類型發(fā)生了變化 — 例如,用戶插入了一個外置鍵盤。 'keyboardHidden' 鍵盤無障礙功能發(fā)生了變化 — 例如,用戶顯示了硬件鍵盤。 'navigation' 導航類型(軌跡球/方向鍵)發(fā)生了變化。(這種情況通常永遠不會發(fā)生。)
以上就是Android如何實現(xiàn)藍牙配對連接功能的詳細內(nèi)容,更多關于Android 藍牙配對連接功能的資料請關注好吧啦網(wǎng)其它相關文章!
相關文章:
1. asp判斷某個文件是否存在的函數(shù)2. Android table布局開發(fā)實現(xiàn)簡單計算器3. IntelliJ IDEA安裝插件的方法步驟4. 在IDEA中實現(xiàn)同時運行2個相同的java程序5. Java如何基于反射機制獲取不同的類6. Vue封裝一個TodoList的案例與瀏覽器本地緩存的應用實現(xiàn)7. 理解PHP5中static和const關鍵字8. Vuex localStorage的具體使用9. ASP.NET泛型三之使用協(xié)變和逆變實現(xiàn)類型轉換10. .Net Core使用Coravel實現(xiàn)任務調(diào)度的完整步驟

網(wǎng)公網(wǎng)安備