Android如何實現藍牙配對連接功能
Android藍牙部分是很復雜的,也涉及很多名詞和功能。本文介紹的配對連接方法適用于一般的藍牙耳機、音響等,并不是連接藍牙 BLE 或者想用藍牙來進行 Socket 通信的。
先來介紹幾種名稱:
Profile:
Bluetooth 的一個很重要特性,就是所有的 Bluetooth 產品都無須實現全部的 Bluetooth 規范。為了更容易的保持 Bluetooth 設備之間的兼容,Bluetooth 規范中定義了 Profile。Profile 定義了設備如何實現一種連接或者應用,你可以把 Profile 理解為連接層或者應用層協。我們標題中的說的連接其實就是去連接各種 Profile。下面介紹的幾種都是Android 實現了的 Profile。A2dp:
表示藍牙立體聲,和藍牙耳機聽歌有關那些,另還有個Avrcp音頻/視頻遠程控制配置文件,是用來聽歌時暫停,上下歌曲選擇的。Handset、Handfree:
和電話相關,藍牙接聽、掛斷電話。其他:
btservice關于藍牙基本操作的目錄,一切由此開始; hdp藍牙關于醫療方面的應用;hid:人機交互接口,藍牙鼠標鍵盤什么的就是這個了 ;pbap:電話號碼簿訪問協議(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);}
注冊廣播由于藍牙的搜索、配對和連接狀態的改變都是系統通過廣播的方式發出來的,所以需要注冊這些廣播來獲取狀態的改變。
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);搜索
獲取已配對的設備。對于之前已經配對成功的設備,系統會把它的信息存儲在本地。再去調用搜索的時候,系統是不會重新再次發現這個設備的。
Set<BluetoothDevice> pairedDevices = mBluetoothAdapter.getBondedDevices();
搜索設備
mBluetoothAdapter.startDiscovery();
系統發現新的藍牙設備了之后,會通過廣播把這個設備的信息發送出來。所以我們要通過截獲 Action 為BluetoothDevice.ACTION_FOUND的 Intent,并得到設備信息。
BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);配對
重點來來了,做了一系列準備工作,拿到了BluetoothDevice下面就要開始配對連接了。但是坑的地方也在這里,首先藍牙設備必須要先配對成功了再去連接各個不同的 Profile,如果直接去連接有的機型確實也可以連上,但是大部分的都沒反應。然后就是 Android 4.4 API 19 以上才開放配對接口,對于之前的系統我們只能通過反射的方式去獲取接口。
配對
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { //Android 4.4 API 19 以上才開放Bond接口 device.createBond();} else { //API 19 以下用反射調用Bond接口 try {device.getClass().getMethod('connect').invoke(device); } catch (Exception e) {e.printStackTrace(); }}
配對成功會發送廣播BluetoothDevice.ACTION_BOND_STATE_CHANGED
//設備綁定狀態改變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);}
配對的幾種狀態
public static final int BOND_NONE = 10; public static final int BOND_BONDING = 11; public static final int BOND_BONDED = 12;連接
配對(綁定)和連接是兩個不同的過程,配對是指兩個設備發現了對方的存在,可以獲取到對方的名稱、地址等信息,有能力建立起連接。連接是指兩個設備共享了一個 RFCOMM 通道,有能力進行數據互傳。確認綁定上了之后,才能開始連接,連接其實就是連接這個藍牙設備支持的 Profile 。
可以觀察一下設置里面藍牙連接的過程過程,就是先開始配對,配對成功后才開始連接所有支持的 Profile。這一步也是比較坑的地方,網上都沒有詳細的對這一塊說明。我也是在 Setting 的源碼里面翻了半天才找到這塊的邏輯。但是系統應用可以直接調用連接的方法,卻不外開放...
首先我們要提前獲取 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);
當我們收到配對成功的廣播或者確定設備已經配對成功后,我們就要調用 Profile 的connect方法來連接。但是這個方法被 Google 給@hide了。像上面一樣用反射...
try {mA2dpService.getClass().getMethod('connect', BluetoothDevice.class).invoke(mA2dpService, item.getDevice()); } catch (Exception e) {e.printStackTrace(); }
連接成功系統會發送廣播BluetoothA2dp.ACTION_CONNECTION_STATE_CHANGED
BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);int profileState = intent.getIntExtra(BluetoothProfile.EXTRA_STATE, BluetoothProfile.STATE_DISCONNECTED);
連接的幾種狀態
/** 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。我們為了連接不是獲取了 Profile 嗎,在連接完成之后一定要關閉掉他,一直不關閉的話會報錯。
mBluetoothAdapter.closeProfileProxy(BluetoothProfile.A2DP, mA2dpService);mA2dpService = null;
藍牙連接成功了之后系統會通知手機狀態發生了改變,就像切換了橫豎屏一樣,需要重新調用這個 Activity 的生命周期。當時我發現只要我一連接成功,我的界面就閃一下回到初始狀態。就納悶了半天,然后我一直以為是用反射連接導致程序異常重啟了...幾經摸索我發現是沒有設置android:configChanges的緣故。
android:configChanges='keyboard|keyboardHidden|navigation'
其實前面兩個屬性我也早想到了,唯獨最后一個,在官方文檔里面寫的This should never normally happen.我天真的相信了,一直沒試它。最后實在沒辦法了把所有的屬性都寫上去,然后一個個減,最終發現了這三少一個都不行。
值 說明 'keyboard' 鍵盤類型發生了變化 — 例如,用戶插入了一個外置鍵盤。 'keyboardHidden' 鍵盤無障礙功能發生了變化 — 例如,用戶顯示了硬件鍵盤。 'navigation' 導航類型(軌跡球/方向鍵)發生了變化。(這種情況通常永遠不會發生。)
以上就是Android如何實現藍牙配對連接功能的詳細內容,更多關于Android 藍牙配對連接功能的資料請關注好吧啦網其它相關文章!
相關文章:
1. SpringBoot+SpringCache實現兩級緩存(Redis+Caffeine)2. ASP.NET MVC視圖頁使用jQuery傳遞異步數據的幾種方式詳解3. php讀取xml中某個元素的內容(PHP5以上才支持)4. Idea 2019.3 本應該搜索到的插件卻搜索不到的解決方法5. ASP.NET泛型三之使用協變和逆變實現類型轉換6. IntelliJ IDEA導入jar包的方法7. 在JSP中使用formatNumber控制要顯示的小數位數方法8. 每日六道java新手入門面試題,通往自由的道路9. 每日六道java新手入門面試題,通往自由的道路--多線程10. 低版本IE正常運行HTML5+CSS3網站的3種解決方案
