最近工作有个小任务是要监测是否有蓝牙设备插入的,具体功能是,若有蓝牙设备插入到机顶盒中,显示蓝牙图标,若蓝牙设备拔出,蓝牙图标消失;因为自己接触android时间没有多长,开始弄这个还有点蒙,所以自己就是一点一点分析,若实现监听蓝牙设备的操作,底层肯定有个东西是关于蓝牙的,就跟底层的接触了下,跟我说是执行shell命令,输入“busybox lsusb”,获取输出内容,输出得到的是当前device上的usb节点信息(自己是这样理解的,大致是这样),每一行都是一个usb的信息,其中某个就是蓝牙的结点。展示例图:
说到这里,想必应该知道根据ID的比较,可以获取蓝牙监听的状态。知道底层的实现方案,那么,前端如何实现呢?
首先,自己分析一下,实现功能需要什么?要一直检测是不是有蓝牙设备插入,要有一个线程,一直在后台运行,这个线程就是用来执行命令行命令,并根据底层提供信息判断是不是有蓝牙设备插入的。其次,实现监测蓝牙设备监测之后还要通知ui线程实现界面上的变化。这里就是蓝牙图标的显示问题了。知道自己要干什么,就可以开始着手了。
首先实现线程一直在后台运行,我用的是“HandlerThread”,之前自己只是看过别人的代码,自己没有用过实现功能,所以这次自己又去翻了一下别人的代码,这次自己使用不得不说还是有收获的。先上代码:
/* 开启定时每三秒检测一次蓝牙设备是否接入,开机即需开始检测,在MainActivity中调用 */
public voidstartCheckUsbWifi() {
stopCheckBlueTooth(); //第一次启用线程的时候先停止检测
initCheckBlueToothThread(); //开始蓝牙设备是否接入监测线程
mCheckBlueToothHandler.sendEmptyMessage(MSG_CHECK_USB_WIFI); //往handler的looper中加入消息,我的理解是第一次启用线,之后会根据handlermesssge中的sendEmptyMessageDelayed方法,每3秒时间执行一次监测的线程
}
/* 开始蓝牙设备是否接入检测线程 */
private voidinitCheckBlueToothThread() {
mCheckBlueToothThread=newHandlerThread("check-blue-tooth"); //参数是线程的名字
mCheckBlueToothThread.start();
mCheckBlueToothHandler=newHandler(mCheckBlueToothThread.getLooper()) {
@Override
public voidhandleMessage(Message msg) {
checkBlueToothState(); //执行蓝牙接入监测
mCheckBlueToothHandler.sendEmptyMessageDelayed(MSG_CHECK_USB_WIFI,CHECK_SERIAL_PERIOD);
}
};
}
/* 执行蓝牙设备接入检测,判断底层节点数据 */
private voidcheckBlueToothState() {
Log.d(TAG,"checkBlueToothState: ");
String command ="busybox lsusb"; //命令行命令
StringBuffer result =execCmdForOutPut(command); //执行命令行,并获取输入数据
if(result.toString().contains("0bda:b720")){ //是否包含蓝牙结点
setUsbWifiEnabled(true); //检测到蓝牙设备,设置蓝牙是否接入Boolean变量为true
}else{
setUsbWifiEnabled(false);
}
ThreadUtil.runOnUiThread(newRunnable() {
@Override
public voidrun() {
if(mListenerList!=null&&mListenerList.size() >0) {
notifyBlueToothChange();//发生改变时通知ui线程观察者发生改变
}else{
EvLog.d(TAG,"IUsbBlueToothChangeObserver is null, updateBlueToothState failed!");
}
}
});
}
/**
* 执行cmd命令并获得输出内容
*/
private staticStringBufferexecCmdForOutPut(String cmd) {
Runtime run = Runtime.getRuntime();
Process proc =null;
InputStream input =null;
BufferedReader in =null;
StringBuffer buffer =newStringBuffer();
try{
proc = run.exec(cmd);
proc.waitFor();
// 具体信息
input = proc.getInputStream();
in =newBufferedReader(newInputStreamReader(input));
String line ="";
while((line = in.readLine()) !=null) {
buffer.append(line +"\r\n");
}
}catch(IOException e) {
EvLog.e(TAG,"IOException :"+ e.getMessage());
}catch(InterruptedException e) {
EvLog.e(TAG,"InterruptedException:"+ e.getMessage());
}finally{
proc.destroy();
try{
input.close();
in.close();
}catch(IOException e) {
e.printStackTrace();
}
}
returnbuffer;
}
之后就是通过观察者模式通知所有已经注册过的observer的观察者发生变化了,今天的大致总结就是这样。