三、Android智能手機藍牙通信功能開發:BluetoothChat例程分析
1. 概述
Bluetooth 是幾乎現在每部手機標準配備的功能,多用于耳機 mic 等設備與手機的連接,除此之外,還可以多部手機之間建立 bluetooth 通信,本文就通過 SDK 中帶的一個聊天室的例程,來介紹一下 Android 上的 Bluetooth 的開發。
在 Android1.x 的時候,相關 API 非常不完善,還不能簡單的使用 Bluetooth 開發,有一個開源項目可以幫助程序員使用、開發藍牙,支持直接方法 bluetooth 協議棧。在 Android2 以后,框架提供了一些官方 API 來進行藍牙的通信,但目前的程序也比較不完善。本文主要討論 Android2 后的 Bluetooth 通信的 API 使用方法。
首先看聊天室的效果圖:
2. Bluetooth 通信 API 介紹
2.1. Bluetooth 通信過程
2.2. Bluetooth API 的主要方法
BluetoothAdapter 類
BluetoothAdapter.getDefaultAdapter() :得到本地默認的 BluetoothAdapter ,若返回為 null 則表示本地不支持藍牙;
isDiscovering() :返回設備是否正在發現周圍藍牙設備;
cancelDiscovery() :取消正在發現遠程藍牙設備的過程;
startDiscovery() :開始發現過程;
getScanMode() :得到本地藍牙設備的 Scan Mode ;
getBondedDevices() :得到已配對的設備;
isEnabled() :藍牙功能是否啟用。
當發現藍牙功能未啟用時,如下調用設置啟用藍牙:
如果發現當前設備沒有打開對外可見模式,則傳遞 Intent 來調用打開可發現模式,代碼如下:
BluetoothDevice 類,此為對應的遠程藍牙 Device
createRfcommSocketToServiceRecord() :創建該 Device 的 socket 。
BluetoothSocket 類
connect() :請求連接藍牙。
getInputStream() :得到輸入流,用于接收遠程方信息。
getOutputStream() :得到輸出流,發送給遠程方的信息。
close() :關閉藍牙連接。
InputStream 類:
read(byte[]) :以阻塞方式讀取輸入流。
OutputStream 類:
write(byte[]) :將信息寫入該輸出流,發送給遠程。
3. BluetoothChat 例程分析
Google 提供的關于 Bluetooth 開發的例程為 Bluetoothchat ,使用截圖可見本文一開始。除去配置及 ui 定義等文件,主程序文件共三個: BluetoothChat.java 、 BluetoothChatService.java 以及 DeviceListActivity.java ,詳細功能可見下面的描述。
3.1. 整體調用關系序列圖
3.2. BluetoothChat.java
例程的主 Activity 。 onCreate() 得到本地 BluetoothAdapter 設備,檢查是否支持。 onStart() 中檢查是否啟用藍牙,并請求啟用,然后執行 setupChat() 。 setupChat() 中先對界面中的控件進行初始化增加點擊監聽器等,然創建 BluetoothChatService 對象,該對象在整個應用過程中存在,并執行藍牙連接建立、消息發送接受等實際的行為。
3.3. BluetoothChatService.java
public synchronized void start() :
開啟 mAcceptThread 線程,由于樣例程序是僅 2 人的聊天過程,故之前先檢測 mConnectThread 和 mConnectedThread 是否運行,運行則先退出這些線程。
public synchronized void connect(BluetoothDevice device) :
取消 CONNECTING 和 CONNECTED 狀態下的相關線程,然后運行新的 mConnectThread 線程。
public synchronized void connected(BluetoothSocket socket, BluetoothDevice device) :
開啟一個 ConnectedThread 來管理對應的當前連接。之前先取消任意現存的 mConnectThread 、 mConnectedThread 、 mAcceptThread 線程,然后開啟新 mConnectedThread ,傳入當前剛剛接受的 socket 連接。最后通過 Handler 來通知 UI 連接 OK 。
public synchronized void stop() :
停止所有相關線程,設當前狀態為 NONE 。
public void write(byte[] out) :
在 STATE_CONNECTED 狀態下,調用 mConnectedThread 里的 write 方法,寫入 byte 。
private void connectionFailed() :
連接失敗的時候處理,通知 ui ,并設為 STATE_LISTEN 狀態。
private void connectionLost() :
當連接失去的時候,設為 STATE_LISTEN 狀態并通知 ui 。
內部類:
private class AcceptThread extends Thread :
創建監聽線程,準備接受新連接。使用阻塞方式,調用 BluetoothServerSocket.accept() 。提供 cancel 方法關閉 socket 。
private class ConnectThread extends Thread :
這是定義的連接線程,專門用來對外發出連接對方藍牙的請求和處理流程。構造函數里通過 BluetoothDevice.createRfcommSocketToServiceRecord() ,從待連接的 device 產生 BluetoothSocket. 然后在 run 方法中 connect ,成功后調用 BluetoothChatSevice 的 connected() 方法。定義 cancel() 在關閉線程時能夠關閉相關 socket 。
private class ConnectedThread extends Thread :
這個是雙方藍牙連接后一直運行的線程。構造函數中設置輸入輸出流。 Run 方法中使用阻塞模式的 InputStream.read() 循環讀取輸入流, 然后 post 到 UI 線程中更新聊天消息。也提供了 write() 將聊天消息寫入輸出流傳輸至對方,傳輸成功后回寫入 UI 線程。最后 cancel() 關閉連接的 socket 。
3.4. DeviceListActivity.java
該類包含 UI 和操作的 Activity 類,作用是得到系統默認藍牙設備的已配對設備列表,以及搜索出的未配對的新設備的列表。然后提供點擊后發出連接設備請求的功能。
除了 RFCOMM 通信外, Android 上關于 Bluetooth 的還有 SDP 、 GAP 、耳機設備連接等內容,本文還未涉及,將會隨著藍牙相關 API 在新版本中的進一步完善來學習使用。
評論
查看更多