Android如何實(shí)現(xiàn)藍(lán)牙配對(duì)連接功能
Android藍(lán)牙部分是很復(fù)雜的,也涉及很多名詞和功能。本文介紹的配對(duì)連接方法適用于一般的藍(lán)牙耳機(jī)、音響等,并不是連接藍(lán)牙 BLE 或者想用藍(lán)牙來(lái)進(jìn)行 Socket 通信的。
先來(lái)介紹幾種名稱:
Profile:
Bluetooth 的一個(gè)很重要特性,就是所有的 Bluetooth 產(chǎn)品都無(wú)須實(shí)現(xiàn)全部的 Bluetooth 規(guī)范。為了更容易的保持 Bluetooth 設(shè)備之間的兼容,Bluetooth 規(guī)范中定義了 Profile。Profile 定義了設(shè)備如何實(shí)現(xiàn)一種連接或者應(yīng)用,你可以把 Profile 理解為連接層或者應(yīng)用層協(xié)。我們標(biāo)題中的說(shuō)的連接其實(shí)就是去連接各種 Profile。下面介紹的幾種都是Android 實(shí)現(xiàn)了的 Profile。A2dp:
表示藍(lán)牙立體聲,和藍(lán)牙耳機(jī)聽(tīng)歌有關(guān)那些,另還有個(gè)Avrcp音頻/視頻遠(yuǎn)程控制配置文件,是用來(lái)聽(tīng)歌時(shí)暫停,上下歌曲選擇的。Handset、Handfree:
和電話相關(guān),藍(lán)牙接聽(tīng)、掛斷電話。其他:
btservice關(guān)于藍(lán)牙基本操作的目錄,一切由此開(kāi)始; hdp藍(lán)牙關(guān)于醫(yī)療方面的應(yīng)用;hid:人機(jī)交互接口,藍(lán)牙鼠標(biāo)鍵盤(pán)什么的就是這個(gè)了 ;pbap:電話號(hào)碼簿訪問(wèn)協(xié)議(Phonebook Access Profile) ...準(zhǔn)備在 AndroidManifest.xml 添加所需的權(quán)限
<uses-permission android:name='android.permission.BLUETOOTH' /><uses-permission android:name='android.permission.BLUETOOTH_ADMIN' />
打開(kāi)藍(lán)牙
mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();if (!mBluetoothAdapter.isEnabled()) { Intent enableIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE); startActivityForResult(enableIntent, REQUEST_ENABLE_BT);}
注冊(cè)廣播由于藍(lán)牙的搜索、配對(duì)和連接狀態(tài)的改變都是系統(tǒng)通過(guò)廣播的方式發(fā)出來(lái)的,所以需要注冊(cè)這些廣播來(lái)獲取狀態(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);搜索
獲取已配對(duì)的設(shè)備。對(duì)于之前已經(jīng)配對(duì)成功的設(shè)備,系統(tǒng)會(huì)把它的信息存儲(chǔ)在本地。再去調(diào)用搜索的時(shí)候,系統(tǒng)是不會(huì)重新再次發(fā)現(xiàn)這個(gè)設(shè)備的。
Set<BluetoothDevice> pairedDevices = mBluetoothAdapter.getBondedDevices();
搜索設(shè)備
mBluetoothAdapter.startDiscovery();
系統(tǒng)發(fā)現(xiàn)新的藍(lán)牙設(shè)備了之后,會(huì)通過(guò)廣播把這個(gè)設(shè)備的信息發(fā)送出來(lái)。所以我們要通過(guò)截獲 Action 為BluetoothDevice.ACTION_FOUND的 Intent,并得到設(shè)備信息。
BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);配對(duì)
重點(diǎn)來(lái)來(lái)了,做了一系列準(zhǔn)備工作,拿到了BluetoothDevice下面就要開(kāi)始配對(duì)連接了。但是坑的地方也在這里,首先藍(lán)牙設(shè)備必須要先配對(duì)成功了再去連接各個(gè)不同的 Profile,如果直接去連接有的機(jī)型確實(shí)也可以連上,但是大部分的都沒(méi)反應(yīng)。然后就是 Android 4.4 API 19 以上才開(kāi)放配對(duì)接口,對(duì)于之前的系統(tǒng)我們只能通過(guò)反射的方式去獲取接口。
配對(duì)
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { //Android 4.4 API 19 以上才開(kāi)放Bond接口 device.createBond();} else { //API 19 以下用反射調(diào)用Bond接口 try {device.getClass().getMethod('connect').invoke(device); } catch (Exception e) {e.printStackTrace(); }}
配對(duì)成功會(huì)發(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);//收到綁定成功的通知后自動(dòng)連接if (item != null && bondState == BluetoothDevice.BOND_BONDED) { connectDevice(item);}
配對(duì)的幾種狀態(tài)
public static final int BOND_NONE = 10; public static final int BOND_BONDING = 11; public static final int BOND_BONDED = 12;連接
配對(duì)(綁定)和連接是兩個(gè)不同的過(guò)程,配對(duì)是指兩個(gè)設(shè)備發(fā)現(xiàn)了對(duì)方的存在,可以獲取到對(duì)方的名稱、地址等信息,有能力建立起連接。連接是指兩個(gè)設(shè)備共享了一個(gè) RFCOMM 通道,有能力進(jìn)行數(shù)據(jù)互傳。確認(rèn)綁定上了之后,才能開(kāi)始連接,連接其實(shí)就是連接這個(gè)藍(lán)牙設(shè)備支持的 Profile 。
可以觀察一下設(shè)置里面藍(lán)牙連接的過(guò)程過(guò)程,就是先開(kāi)始配對(duì),配對(duì)成功后才開(kāi)始連接所有支持的 Profile。這一步也是比較坑的地方,網(wǎng)上都沒(méi)有詳細(xì)的對(duì)這一塊說(shuō)明。我也是在 Setting 的源碼里面翻了半天才找到這塊的邏輯。但是系統(tǒng)應(yīng)用可以直接調(diào)用連接的方法,卻不外開(kāi)放...
首先我們要提前獲取 Profile,這里拿A2dp來(lái)舉例,其他的原理是一樣的。
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)我們收到配對(duì)成功的廣播或者確定設(shè)備已經(jīng)配對(duì)成功后,我們就要調(diào)用 Profile 的connect方法來(lái)連接。但是這個(gè)方法被 Google 給@hide了。像上面一樣用反射...
try {mA2dpService.getClass().getMethod('connect', BluetoothDevice.class).invoke(mA2dpService, item.getDevice()); } catch (Exception e) {e.printStackTrace(); }
連接成功系統(tǒng)會(huì)發(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;坑坑坑
哈哈,你以為連接上了就完事了嗎?!這里面還有幾個(gè)坑容我給你說(shuō)說(shuō)。
不要忘記關(guān)閉 Profile。我們?yōu)榱诉B接不是獲取了 Profile 嗎,在連接完成之后一定要關(guān)閉掉他,一直不關(guān)閉的話會(huì)報(bào)錯(cuò)。
mBluetoothAdapter.closeProfileProxy(BluetoothProfile.A2DP, mA2dpService);mA2dpService = null;
藍(lán)牙連接成功了之后系統(tǒng)會(huì)通知手機(jī)狀態(tài)發(fā)生了改變,就像切換了橫豎屏一樣,需要重新調(diào)用這個(gè) Activity 的生命周期。當(dāng)時(shí)我發(fā)現(xiàn)只要我一連接成功,我的界面就閃一下回到初始狀態(tài)。就納悶了半天,然后我一直以為是用反射連接導(dǎo)致程序異常重啟了...幾經(jīng)摸索我發(fā)現(xiàn)是沒(méi)有設(shè)置android:configChanges的緣故。
android:configChanges='keyboard|keyboardHidden|navigation'
其實(shí)前面兩個(gè)屬性我也早想到了,唯獨(dú)最后一個(gè),在官方文檔里面寫(xiě)的This should never normally happen.我天真的相信了,一直沒(méi)試它。最后實(shí)在沒(méi)辦法了把所有的屬性都寫(xiě)上去,然后一個(gè)個(gè)減,最終發(fā)現(xiàn)了這三少一個(gè)都不行。
值 說(shuō)明 'keyboard' 鍵盤(pán)類型發(fā)生了變化 — 例如,用戶插入了一個(gè)外置鍵盤(pán)。 'keyboardHidden' 鍵盤(pán)無(wú)障礙功能發(fā)生了變化 — 例如,用戶顯示了硬件鍵盤(pán)。 'navigation' 導(dǎo)航類型(軌跡球/方向鍵)發(fā)生了變化。(這種情況通常永遠(yuǎn)不會(huì)發(fā)生。)
以上就是Android如何實(shí)現(xiàn)藍(lán)牙配對(duì)連接功能的詳細(xì)內(nèi)容,更多關(guān)于Android 藍(lán)牙配對(duì)連接功能的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. python爬蟲(chóng)實(shí)戰(zhàn)之制作屬于自己的一個(gè)IP代理模塊2. asp批量添加修改刪除操作示例代碼3. 基于javaweb+jsp實(shí)現(xiàn)企業(yè)財(cái)務(wù)記賬管理系統(tǒng)4. css代碼優(yōu)化的12個(gè)技巧5. 如何在jsp界面中插入圖片6. Vue element ui用戶展示頁(yè)面的實(shí)例7. Ajax返回值類型與用法實(shí)例分析8. 使用FormData進(jìn)行Ajax請(qǐng)求上傳文件的實(shí)例代碼9. .NET6打包部署到Windows Service的全過(guò)程10. HTML 絕對(duì)路徑與相對(duì)路徑概念詳細(xì)
