Android提高之蓝牙隐藏API探秘

前面文章讲解了Android的蓝牙基本用法,本文讲得深入些,探讨下蓝牙方面的隐藏API。用过Android系统设置(Setting)的人都知道蓝牙搜索之后可以建立配对和解除配对,但是这两项功能的函数没有在SDK中给出,那么如何去使用这两项功能呢?本文利用JAVA的反射机制去调用这两项功能对应的函数:createBond和removeBond,具体的发掘和实现步骤如下:

1.使用Git工具下载platform/packages/apps/Settings.git,在Setting源码中查找关于建立配对和解除配对的API,知道这两个API的宿主(BluetoothDevice);

2.使用反射机制对BluetoothDevice枚举其所有方法和常量,看看是否存在:

static public void printAllInform(Class clsShow) {
 try {
 // 取得所有方法
 Method[] hideMethod = clsShow.getMethods();
 int i = 0;
 for (; i < hideMethod.length; i++) {
  Log.e("method name", hideMethod[i].getName());
 }
 // 取得所有常量
 Field[] allFields = clsShow.getFields();
 for (i = 0; i < allFields.length; i++) {
  Log.e("Field name", allFields[i].getName());
 }
 } catch (SecurityException e) {
 // throw new RuntimeException(e.getMessage());
 e.printStackTrace();
 } catch (IllegalArgumentException e) {
 // throw new RuntimeException(e.getMessage());
 e.printStackTrace();
 } catch (Exception e) {
 // TODO Auto-generated catch block
 e.printStackTrace();
 }
}

结果如下:

11-29 09:19:12.012: method name(452): cancelBondProcess
11-29 09:19:12.020: method name(452): cancelPairingUserInput
11-29 09:19:12.020: method name(452): createBond
11-29 09:19:12.020: method name(452): createInsecureRfcommSocket
11-29 09:19:12.027: method name(452): createRfcommSocket
11-29 09:19:12.027: method name(452): createRfcommSocketToServiceRecord
11-29 09:19:12.027: method name(452): createScoSocket
11-29 09:19:12.027: method name(452): describeContents
11-29 09:19:12.035: method name(452): equals
11-29 09:19:12.035: method name(452): fetchUuidsWithSdp
11-29 09:19:12.035: method name(452): getAddress
11-29 09:19:12.035: method name(452): getBluetoothClass
11-29 09:19:12.043: method name(452): getBondState
11-29 09:19:12.043: method name(452): getName
11-29 09:19:12.043: method name(452): getServiceChannel
11-29 09:19:12.043: method name(452): getTrustState
11-29 09:19:12.043: method name(452): getUuids
11-29 09:19:12.043: method name(452): hashCode
11-29 09:19:12.043: method name(452): isBluetoothDock
11-29 09:19:12.043: method name(452): removeBond
11-29 09:19:12.043: method name(452): setPairingConfirmation
11-29 09:19:12.043: method name(452): setPasskey
11-29 09:19:12.043: method name(452): setPin
11-29 09:19:12.043: method name(452): setTrust
11-29 09:19:12.043: method name(452): toString
11-29 09:19:12.043: method name(452): writeToParcel
11-29 09:19:12.043: method name(452): convertPinToBytes
11-29 09:19:12.043: method name(452): getClass
11-29 09:19:12.043: method name(452): notify
11-29 09:19:12.043: method name(452): notifyAll
11-29 09:19:12.043: method name(452): wait
11-29 09:19:12.051: method name(452): wait
11-29 09:19:12.051: method name(452): wait

3.如果枚举发现API存在(SDK却隐藏),则自己实现调用方法:

/**
 * 与设备配对 参考源码:platform/packages/apps/Settings.git
 * /Settings/src/com/android/settings/bluetooth/CachedBluetoothDevice.java
 */
static public boolean createBond(Class btClass,BluetoothDevice btDevice) throws Exception {
 Method createBondMethod = btClass.getMethod("createBond");
 Boolean returnValue = (Boolean) createBondMethod.invoke(btDevice);
 return returnValue.booleanValue();
}
/**
 * 与设备解除配对 参考源码:platform/packages/apps/Settings.git
 * /Settings/src/com/android/settings/bluetooth/CachedBluetoothDevice.java
 */
static public boolean removeBond(Class btClass,BluetoothDevice btDevice) throws Exception {
 Method removeBondMethod = btClass.getMethod("removeBond");
 Boolean returnValue = (Boolean) removeBondMethod.invoke(btDevice);
 return returnValue.booleanValue();
}

此处注意:SDK之所以不给出隐藏的API肯定有其原因,也许是出于安全性或者是后续版本兼容性的考虑,因此不能保证隐藏API能在所有Android平台上很好地运行。

本文程序运行效果如下图所示:

main.xml源码如下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
 android:orientation="vertical" android:layout_width="fill_parent"
 android:layout_height="fill_parent">
 <LinearLayout android:id="@+id/LinearLayout01"
 android:layout_height="wrap_content" android:layout_width="fill_parent">
 <Button android:layout_height="wrap_content" android:id="@+id/btnSearch"
  android:text="Search" android:layout_width="160dip"></Button>
 <Button android:layout_height="wrap_content"
  android:layout_width="160dip" android:text="Show" android:id="@+id/btnShow"></Button>
 </LinearLayout>
 <LinearLayout android:id="@+id/LinearLayout02"
 android:layout_width="wrap_content" android:layout_height="wrap_content"></LinearLayout>
 <ListView android:id="@+id/ListView01" android:layout_width="fill_parent"
 android:layout_height="fill_parent">
 </ListView>
</LinearLayout>

工具类ClsUtils.java源码如下:

package com.testReflect;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import android.bluetooth.BluetoothDevice;
import android.util.Log;
public class ClsUtils {
 /**
 * 与设备配对 参考源码:platform/packages/apps/Settings.git
 * /Settings/src/com/android/settings/bluetooth/CachedBluetoothDevice.java
 */
 static public boolean createBond(Class btClass,BluetoothDevice btDevice) throws Exception {
 Method createBondMethod = btClass.getMethod("createBond");
 Boolean returnValue = (Boolean) createBondMethod.invoke(btDevice);
 return returnValue.booleanValue();
 }
 /**
 * 与设备解除配对 参考源码:platform/packages/apps/Settings.git
 * /Settings/src/com/android/settings/bluetooth/CachedBluetoothDevice.java
 */
 static public boolean removeBond(Class btClass,BluetoothDevice btDevice) throws Exception {
 Method removeBondMethod = btClass.getMethod("removeBond");
 Boolean returnValue = (Boolean) removeBondMethod.invoke(btDevice);
 return returnValue.booleanValue();
 }
 /**
 *
 * @param clsShow
 */
 static public void printAllInform(Class clsShow) {
 try {
  // 取得所有方法
  Method[] hideMethod = clsShow.getMethods();
  int i = 0;
  for (; i < hideMethod.length; i++) {
  Log.e("method name", hideMethod[i].getName());
  }
  // 取得所有常量
  Field[] allFields = clsShow.getFields();
  for (i = 0; i < allFields.length; i++) {
  Log.e("Field name", allFields[i].getName());
  }
 } catch (SecurityException e) {
  // throw new RuntimeException(e.getMessage());
  e.printStackTrace();
 } catch (IllegalArgumentException e) {
  // throw new RuntimeException(e.getMessage());
  e.printStackTrace();
 } catch (Exception e) {
  // TODO Auto-generated catch block
  e.printStackTrace();
 }
 }
}

主程序testReflect.java的源码如下:

package com.testReflect;
import java.util.ArrayList;
import java.util.List;
import android.app.Activity;
import android.bluetooth.BluetoothAdapter;
import android.bluetooth.BluetoothDevice;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.AdapterView;
import android.widget.ArrayAdapter;
import android.widget.Button;
import android.widget.ListView;
import android.widget.Toast;
public class testReflect extends Activity {
 Button btnSearch, btnShow;
 ListView lvBTDevices;
 ArrayAdapter<String> adtDevices;
 List<String> lstDevices = new ArrayList<String>();
 BluetoothDevice btDevice;
 BluetoothAdapter btAdapt;
 @Override
 public void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.main);
 btnSearch = (Button) this.findViewById(R.id.btnSearch);
 btnSearch.setOnClickListener(new ClickEvent());
 btnShow = (Button) this.findViewById(R.id.btnShow);
 btnShow.setOnClickListener(new ClickEvent());
 lvBTDevices = (ListView) this.findViewById(R.id.ListView01);
 adtDevices = new ArrayAdapter<String>(testReflect.this,
  android.R.layout.simple_list_item_1, lstDevices);
 lvBTDevices.setAdapter(adtDevices);
 lvBTDevices.setOnItemClickListener(new ItemClickEvent());

 btAdapt = BluetoothAdapter.getDefaultAdapter();// 初始化本机蓝牙功能
 if (btAdapt.getState() == BluetoothAdapter.STATE_OFF)// 开蓝牙
  btAdapt.enable();
 // 注册Receiver来获取蓝牙设备相关的结果
 IntentFilter intent = new IntentFilter();
 intent.addAction(BluetoothDevice.ACTION_FOUND);
 intent.addAction(BluetoothDevice.ACTION_BOND_STATE_CHANGED);
 registerReceiver(searchDevices, intent);

 }
 private BroadcastReceiver searchDevices = new BroadcastReceiver() {
 public void onReceive(Context context, Intent intent) {
  String action = intent.getAction();
  Bundle b = intent.getExtras();
  Object[] lstName = b.keySet().toArray();
  // 显示所有收到的消息及其细节
  for (int i = 0; i < lstName.length; i++) {
  String keyName = lstName[i].toString();
  Log.e(keyName, String.valueOf(b.get(keyName)));
  }
  // 搜索设备时,取得设备的MAC地址
  if (BluetoothDevice.ACTION_FOUND.equals(action)) {
  BluetoothDevice device = intent
   .getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
  if (device.getBondState() == BluetoothDevice.BOND_NONE) {
   String str = "未配对|" + device.getName() + "|" + device.getAddress();
   lstDevices.add(str); // 获取设备名称和mac地址
   adtDevices.notifyDataSetChanged();
  }
  }
 }
 };
 class ItemClickEvent implements AdapterView.OnItemClickListener {

 @Override
 public void onItemClick(AdapterView<?> arg0, View arg1, int arg2,
  long arg3) {
  btAdapt.cancelDiscovery();
  String str = lstDevices.get(arg2);
  String[] values = str.split("//|");
  String address=values[2];
  btDevice = btAdapt.getRemoteDevice(address);
  try {
  if(values[0].equals("未配对"))
  {
   Toast.makeText(testReflect.this, "由未配对转为已配对", 500).show();
   ClsUtils.createBond(btDevice.getClass(), btDevice);
  }
  else if(values[0].equals("已配对"))
  {
   Toast.makeText(testReflect.this, "由已配对转为未配对", 500).show();
   ClsUtils.removeBond(btDevice.getClass(), btDevice);
  }
  } catch (Exception e) {
  // TODO Auto-generated catch block
  e.printStackTrace();
  }
 }
 }
 /**
 * 按键处理
 * @author GV
 *
 */
 class ClickEvent implements View.OnClickListener {
 @Override
 public void onClick(View v) {
  if (v == btnSearch) {//搜索附近的蓝牙设备
  lstDevices.clear();
  Object[] lstDevice = btAdapt.getBondedDevices().toArray();
  for (int i = 0; i < lstDevice.length; i++) {
   BluetoothDevice device=(BluetoothDevice)lstDevice[i];
   String str = "已配对|" + device.getName() + "|" + device.getAddress();
   lstDevices.add(str); // 获取设备名称和mac地址
   adtDevices.notifyDataSetChanged();
  }
  // 开始搜索
  setTitle("本机蓝牙地址:" + btAdapt.getAddress());
  btAdapt.startDiscovery();
  }
  else if(v==btnShow){//显示BluetoothDevice的所有方法和常量,包括隐藏API
  ClsUtils.printAllInform(btDevice.getClass());
  }
 }
 }
}

希望本文实例能够对大家进行Android程序开发有一定的借鉴帮助作用。

(0)

相关推荐

  • Android开发学习笔记之通过API接口将LaTex数学函数表达式转化为图片形式

    本文将讲解如何通过codecogs.com和Google.com提供的API接口来将LaTeX数学函数表达式转化为图片形式.具体思路如下: (1)通过EditText获取用户输入的LaTeX数学表达式,然后对表达式格式化使之便于网络传输. (2)将格式化之后的字符串,通过Http请求发送至codecogs.com或者Google.com. (3)获取网站返回的数据流,将其转化为图片,并显示在ImageView上. 具体过程为: 1.获取并格式化LaTeX数学表达式 首先,我们在这个网站输入LaT

  • android 微信 sdk api调用不成功解决方案

    最近一直在调用微信的API,却发现一直调用不成功,纠结了好久,各方面找教程,找官方,官方里的文档也只是写得很模糊,说是按三步走. 1.申请App_ID 2.填写包名3. 获取程序签名的md5值, 这三步只要你走对了就能调通,可是大家都不知道有时候我们打包的keystore和我们打包的keystore获取到的程序签名的md5是不一样的.我们在申请的时候填的程序签名值是正式打包的,但我们在eclipse部署上去的却是用的我们默认的debug.keystore.而这样导致的后果就是程序签名不一样,会一

  • 使用android隐藏api实现亮度调节的方法

    Android 1.5之后将调节亮度的api隐藏了,不过android的源码中依然有方法可以实现,如下: 复制代码 代码如下: IPowerManager power = IPowerManager.Stub.asInterface(ServiceManager .getService("power")); if (power != null) { power.setBacklightBrightness(brightness); 上面的代码无法直接在官方的sdk中调用,会提示找不到I

  • Android 高版本API方法在低版本系统上的兼容性处理

    Android 版本更替,新的版本带来新的特性,新的方法. 新的方法带来许多便利,但无法在低版本系统上运行,如果兼容性处理不恰当,APP在低版本系统上,运行时将会crash. 本文以一个具体的例子说明如何在使用高API level的方法时处理好兼容性问题. 例子:根据给出路径,获取此路径所在分区的总空间大小. 在安卓中的文件存储使用参考中提到: 获取文件系统用量情况,在API level 9及其以上的系统,可直接调用File对象的相关方法,以下需自行计算 一般实现 就此需求而言,API leve

  • android通过google api获取天气信息示例

    android通过google API获取天气信息 复制代码 代码如下: public class WeatherActivity extends Activity { private TextView txCity; private Button btnSearch; private Handler weatherhandler; private Dialog progressDialog; private Timer timer;    /** Called when the activit

  • android开发教程之获取使用当前api的应用程序名称

    比如要获取打开摄像头的应用程序名称,只需要在frameworks/base/core/android/hardware/Camera.java中open()方法中加上如下代码就可以了. 复制代码 代码如下: Application application = ActivityThread.currentApplication();if (application != null) {    String packageName = application.getPackageName();   

  • android通过Location API显示地址信息的实现方法

    本文实例讲述了android通过Location API显示地址信息的实现方法.分享给大家供大家参考.具体如下: android的Locatin API,可以通过Geocoder类,显示具体经纬度的地址信息.如: 通过Geocoder的方法getFromLocation()可以得到Address对象的List.我只取一个Address结果,可以取多个,但是意义不大. StringBuilder builder = new StringBuilder(); builder.append("北纬:&

  • android monkey自动化测试改为java调用monkeyrunner Api

    众所周知,一般情况下我们使用android中的monkeyrunner进行自动化测试时,使用的是python语言来写测试脚本.不过,最近发现可以用java调用monkeyrunner Api,用java语言写测试脚本. 于是,就简单研究了一下.这里做一些总结.希望有对在研究的午饭可以有所用处. 开始时,搜素到一些零碎的教程,说使用java调用monkeyrunner时,需要导入android sdk  tools路径下的lib里面的4个包:ddmlib.jar,guavalib.jar,monk

  • Android提高之蓝牙隐藏API探秘

    前面文章讲解了Android的蓝牙基本用法,本文讲得深入些,探讨下蓝牙方面的隐藏API.用过Android系统设置(Setting)的人都知道蓝牙搜索之后可以建立配对和解除配对,但是这两项功能的函数没有在SDK中给出,那么如何去使用这两项功能呢?本文利用JAVA的反射机制去调用这两项功能对应的函数:createBond和removeBond,具体的发掘和实现步骤如下: 1.使用Git工具下载platform/packages/apps/Settings.git,在Setting源码中查找关于建立

  • Android提高之蓝牙传感应用实例

    前面文章介绍了Android利用麦克风采集并显示模拟信号的实现方法,这种采集手段适用于无IO控制.单纯读取信号的情况.如果传感器本身需要包含控制电路(例如采集血氧信号需要红外和红外线交替发射),那么传感器本身就需要带一片主控IC,片内采集并输出数字信号了.Android手机如何在不改硬件电路的前提下与这类数字传感器交互呢?可选的通信方式就有USB和蓝牙,两种方式各有好处:USB方式可以给传感器供电,蓝牙方式要自备电源:USB接口标准不一,蓝牙普遍支持SPP协议.本文就选择蓝牙方式做介绍,介绍An

  • Android提高之TelephonyManager功能探秘

    前面文章介绍了如何使用JAVA的反射机制来调用蓝牙的隐藏API,本文继续来练习JAVA的反射机制,探秘TelephonyManager在Framework里包含却在SDK隐藏的几项功能. 先来看一下本文程序运行的效果图,如下所示: 本文程序演示了以下功能: 1.所有来电自动接听: 2.所有来电自动挂断: 3.开启/关闭Radio: 4.开启/关闭数据连接(WAP or NET的连接). 调用TelephonyManager的隐藏API是先参考Framework的/base/telephony/j

  • Android提高Service优先级的方法分析

    本文实例讲述了Android提高Service优先级的方法.分享给大家供大家参考,具体如下: Android 系统对于内存管理有自己的一套方法,为了保障系统有序稳定的运信,系统内部会自动分配,控制程序的内存使用.当系统觉得当前的资源非常有限的时候,为了保 证一些优先级高的程序能运行,就会杀掉一些他认为不重要的程序或者服务来释放内存.这样就能保证真正对用户有用的程序仍然再运行.如果你的 Service 碰上了这种情况,多半会先被杀掉.但如果你增加 Service 的优先级就能让他多留一会,我们可以

  • Android如何实现蓝牙配对连接功能

    目录 本文适用的范围 准备 搜索 配对 连接 坑坑坑 本文适用的范围 Android蓝牙部分是很复杂的,也涉及很多名词和功能.本文介绍的配对连接方法适用于一般的蓝牙耳机.音响等,并不是连接蓝牙 BLE 或者想用蓝牙来进行 Socket 通信的. 先来介绍几种名称: Profile: Bluetooth 的一个很重要特性,就是所有的 Bluetooth 产品都无须实现全部的 Bluetooth 规范.为了更容易的保持 Bluetooth 设备之间的兼容,Bluetooth 规范中定义了 Profi

  • 推荐8项提高 ASP.NET Web API 性能的技术

    在本文中,我将介绍8项提高 ASP.NET Web API 性能的技术. 1) 使用最快的 JSON 序列化工具 JSON 的序列化对整个 ASP.NET Web API 的性能有着关键性的影响.在我的一个项目里,我从JSON.NET 序列化工具转到了ServiceStack.Text有一年半了. 我测量过,Web API 的性能提升了20%左右.我强烈建议你去尝试一下这个序列化工具.这里有一些最近的流行序列化工具性能的比较数据. 来源:theburningmonk 更新: 似乎It seams

  • Android ListView自动显示隐藏布局的实现方法

    借助View的OnTouchListener接口来监听listView的滑动,通过比较与上次坐标的大小,判断滑动方向,并通过滑动方向来判断是否需显示或者隐藏对应的布局,并且带有动画效果. 1.自动显示隐藏Toolbar 首先给listView增加一个HeaderView,避免第一个Item被Toolbar遮挡. View header=new View(this); header.setLayoutParams(new AbsListView.LayoutParams( AbsListView.

  • Android编程之蓝牙测试实例

    本文实例讲述了Android编程之蓝牙测试.分享给大家供大家参考.具体分析如下: 一.软件平台: win7 + eclipse + sdk 二.设计思路: 配合倒计时定时器实现蓝牙打开,可见,扫描三个功能 三.源代码: main.xml: <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/re

  • Android开发之完全隐藏软键盘的方法

    隐藏软键盘一直是我头痛的事情,没有找到一种真正能隐藏的方法.点击EditText的时候总是弹出软键盘.-----杯具 杯具(一): InputMethodManager im =(InputMethodManager) mEdit getContext() .getSystemService(Context.INPUT_METHOD_SERVICE); im.hideSoftInputFromWindow(SoftKeyTest.this.getCurrentFocus().getWindowT

随机推荐