- 通知; 2、對話框; 3、消息提示框
- 在使用Android藍牙適配器中的startDiscovery需要先打開定位服務
- 在連接藍牙之前,還要先取消藍牙設備的掃描,否則容易連接失敗
前段時間自己學做了一個基于安卓經典藍牙的app,也遇到不少坑,在網上相對與BLE低功耗藍牙來說,它的資料方面都是比較缺少的,而且網上經典藍牙源碼大部分都很長,對于入門的同學來說,看沒兩分鐘就產生放棄的念頭了。 第一步:在AndroidManifest中添加: <!--管理藍牙需要-->
<uses-permission android:name="android.permission.BLUETOOTH" />
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />
<!--搜索藍牙需要,因為藍牙可以被用來定位,所以需要定位權限-->
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/> 上面都是我們使用到設備功能的權限申請。 第二步:設置兩個按鈕分別是打開和關閉藍牙 對藍牙設備干事情,那么我們就得拿到它的接口來控制它 //獲取藍牙適配器
private BluetoothAdapter mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter(); 打開藍牙: 1、 public void OpenBlueTooth(View view) {
if(isSupportBlueTooth() == true) {
if(!BlueToothState()) {
Intent intent = new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE);
startActivityForResult(intent,0);
showToast("親,打開了噢!需要什么幫助嗎?");
}
else {
showToast("親,已經打開了噢,無需重復打開。");
}
}
else {
showToast("親,您不支持此藍牙設備!");
}
} 2、打開藍牙應該先確定使用的手機是否支持藍牙設備(手機沒有就沒必要進行下一步了) //判斷是否支持藍牙
public boolean isSupportBlueTooth() {
if(mBluetoothAdapter != null) {
return true;
}
else {
return false;
}
} 3、然后打開前還需要知道手機是否打開藍牙了吧!那就獲取手機現在藍牙設備的狀態 //獲取藍牙狀態
public boolean BlueToothState() {
assert (mBluetoothAdapter != null); //若不支持該藍牙設備會有個斷言
return mBluetoothAdapter.isEnabled();
} 2、3小步驟都只是個函數功能體,都是為1、步驟使用的,初學者會疑惑什么是斷言,就是做判斷,真的那么下面就不執行咯!還有下面是是一個消息對象, 獲取用戶對藍牙適配器申請打開的請求消息,使用startActivityForResult對消息做出應答,會有對話框的提示效果 Intent intent = new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE);
startActivityForResult(intent,0); 關閉藍牙: public void CloseBlueTooth(View view) {
mBluetoothAdapter.disable();
showToast("親,我們會再見面的!");
} 先說明一點我這里的還是都在有View view形參接口,這是一個動作的監聽接口,下圖有設置的圖片,這里設置就可以了。
感覺打開藍牙和關閉藍牙挺簡單的吧! 難度開始來了噢!
第三步: 打開了藍牙我們就得弄一個來搜索周圍的設備吧! 下面是一個定義好的搜索藍牙函數 public void startDiscovery() {
if (mBluetoothAdapter.isDiscovering()) {mBluetoothAdapter.cancelDiscovery(); Toast.makeText(this,"搜索器打開",Toast.LENGTH_SHORT).show();}
mBluetoothAdapter.startDiscovery();
if (!mBluetoothAdapter.isDiscovering()) {Toast.makeText(this,"搜索器沒打開",Toast.LENGTH_SHORT).show();}
} 有點看不懂嗎,其實還好吧,就是第一次判斷是否開啟搜索周圍藍牙設備,開啟了就把它關了,重新打開在搜索,在判斷我是否開啟了搜索沒。(isDiscovery是開啟搜索,cancelDiscovery是關閉搜索的意思) 有搜索藍牙設備函數了,我們就搞一個按鍵讓其有觸發效果吧! //按鈕搜尋藍牙
Button button_discovery = (Button) findViewById(R.id.DiscoveryBT);
button_discovery.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
startDiscovery();
}
}); 開啟了搜索藍牙設備我們得有東西接收這些設備吧,不然它偷偷摸摸搜索鬼知道它搜了什么東西。那我們就得用廣播把這些偷偷摸摸搜索到了設備把它弄出來,放進“箱子”里。 這就是我們定義的廣播接收器 public class MyReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
if(intent.getAction().equals(BluetoothAdapter.ACTION_DISCOVERY_STARTED)) {
Toast.makeText(context, "開始", Toast.LENGTH_SHORT).show();
} else if (BluetoothDevice.ACTION_FOUND.equals(intent.getAction())) {
BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
//////////////////////////////////////////////創建搜索藍牙列表的///////////////////////////////////////////////////////////////////
for (int i = 0; i < deviceAdress.size(); i++) {
if (deviceAdress.get(i).getAddress().equals(device.getAddress())) return;
//上面if語句就是去除已經獲取的藍牙設備
}
// 不是重復的就添加到列表中(獲取未配對的藍牙設備)
deviceAdress.add(device); //添加地址到列表中 用于鑒別是否已經添加列表和點擊事件用的
deviceName.add("地址:"+device.getAddress()+"\n"+"名稱:"+device.getName()); //存放藍牙名稱和地址用于顯示到列表上的
} else if(intent.getAction().equals(BluetoothAdapter.ACTION_DISCOVERY_FINISHED)) {
}
}
} 代碼都有注釋我就不一一說明了,說一下箱子要在外面定義 public ArrayList<BluetoothDevice> deviceAdress = new ArrayList<>(); //存放藍牙設備(這里Adress我忘了改過來了,這是存放設備不是設備地址)
public ArrayList<String> deviceName = new ArrayList<>(); //存放藍牙名稱和地址 然后就是用廣播類里面用if那些進行信息篩選(用藍牙設備這個方法獲取設備的信息BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE) 然后就是把搜索到的device 你可以用Toast.make()打印一下信息看看效果 記得廣播接收器的使用一定要注冊廣播接收器,不然就廢掉了 IntentFilter intentFilter = new IntentFilter();
intentFilter.addAction(BluetoothDevice.ACTION_FOUND);//發現藍牙
intentFilter.addAction(BluetoothAdapter.ACTION_DISCOVERY_FINISHED);//搜索停止
intentFilter.addAction(BluetoothAdapter.ACTION_DISCOVERY_STARTED );//搜索啟動
//上面是添加動作事件
//注冊廣播接收者的對象
registerReceiver(myReceiver,intentFilter); 都放在箱子里了,我們應該把它拿出來一個一個展示一下吧! 這展示的方法就是用列表把它展示出來: public ListView listView; //定義展示列表
//////////////////////////////////////////////創建搜索藍牙列表的///////////////////////////////////////////////////////////////////并把
arrayAdapter = new ArrayAdapter(this, android.R.layout.simple_expandable_list_item_1,deviceName); //實例化ArrayAdapter對象deviceName集合數據放入arrayAdapter適配器集合內
listView = (ListView) findViewById(R.id.list); //獲取列表框的
listView.setAdapter(arrayAdapter); //將arrayAdapter集合內的數據加載到列表框 就是適配器對象與ListView關聯 這樣放進列表你以為就可以了嗎,你怎么知道它加載了多少個已發現的設備呢?這需要我們到廣播接收器這個線程中間斷刷新載入新設備 arrayAdapter.notifyDataSetChanged(); //更新列表 下面是我定義的一個刷新列表搜索 Button button= (Button) findViewById(R.id.Broadcast); //獲取布局文件中的廣播按鈕
button.setOnClickListener(new View.OnClickListener() { //為按鈕設置單擊事件
@Override
public void onClick(View v) {
//////////藍牙刷新///////////////
deviceAdress.clear(); //
deviceName.clear(); //
startDiscovery(); //
////////////////////////////////
}
}); 到現在你就可以試試你做的可不可以將搜索到的藍牙加載到列表中顯示了,可以的化,看看可不可以刷新列表。
第四步: 把我們列表中的設備能進行點擊觸發 //連接藍牙
public void Connect_BT(ArrayList<BluetoothDevice> deviceAdress) {
//MainActivity 實現OnItemClickListener 然后重寫方法
listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
BluetoothDevice clickDevice = (BluetoothDevice)deviceAdress.get(position);
String s1 = String.valueOf(position); //編號
Toast.makeText(MainActivity.this, s1 + "--" + clickDevice.getName() + "--" + clickDevice.getAddress(),Toast.LENGTH_SHORT).show();
//在連接前需要先關閉搜索
}
});
} 把我們列表按鍵觸發這個類放到廣播接收器這個線程中 Connect_BT(deviceAdress); 按下列表就有這個提示框咯!開心吧
第五步: 連接我們選中的藍牙 連接這些藍牙都需要協議的,都是廠家底層工作人員定好的 //手機連接的UUID
//設備連接的UUID由廠商決定。
private final String BLUETOOTH_UUID = "00001101-0000-1000-8000-00805F9B34FB"; //藍牙通信的UUID,必須為這個,如果換成其他的UUID會無法通信
private BluetoothSocket bluetoothSocket = null; UUID我就步詳細說了,你們自行百度。BluetoothSocket套接字就是連接藍牙服務器用的。 在Connect_BT添加下面代碼即可() //在連接前需要先關閉搜索
//點擊列表,去請求服務器
if (clickDevice != null) {
new Thread(new Runnable() {
@Override
public void run() {
try {
bluetoothSocket = clickDevice.createRfcommSocketToServiceRecord(UUID.fromString(BLUETOOTH_UUID));
bluetoothSocket.connect();
} catch (IOException e) {
e.printStackTrace();
}
}
}).start();
} 這我要說明一下connect方法會阻塞線程,我們需要在下面創建一個子線程來運行,不然程序奔潰別找我就可以了。 到這我們就所有基本弄好了
第五步: 接收和發送數據 這里我略微講一下 // 用來收數據
InputStream inputStream = bluetoothSocket.getInputStream();
// 用來發數據
OutputStream outputStream; 下面是我用按鍵發送數據的代碼(參考) Button button_oc3 = (Button) findViewById(R.id.button_open_close3);
button_oc3.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
String text = "d12";
try {
outputStream = bluetoothSocket.getOutputStream();
outputStream.write(text.getBytes());
outputStream.flush();
} catch (IOException e) {
e.printStackTrace();
}
}
}); 記住完成BluetoothSocket的一系列操作后,記住用close()方法關閉套接字噢,不然手機很耗電,燒壞別找我麻煩。 接收數據的話會比較麻煩一點點,需要用一個線程和while死循環繼續時刻接收(因為InputStream 也會阻塞線程的)
到這就告一段落了。 我有一個老師是這樣說過的游戲打累了就休息一下,休息的方法就是敲代碼。
在社會學到的要學會分享回報社會
(有什么好的想法和建議可以分享一下給我) 廣州大學華軟軟件學院djw
51hei.png (4.58 KB, 下載次數: 137)
下載附件
2020-11-28 17:37 上傳
全部資料51hei下載地址(源碼):
經典藍牙開發過程.docx
(148.04 KB, 下載次數: 56)
2020-11-28 17:15 上傳
點擊文件名下載附件
下載積分: 黑幣 -5
|