HardBirch

Android提高第十三篇之探秘蓝牙隐藏API

时间:10-11-29 栏目:安卓入门与提高 作者:张飞不张,文采横飞 评论:70 点击: 24,629 次

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

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

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

 
























  

结果如下:

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 removeBond(Class btClass,BluetoothDevice btDevice) throws Exception {
Method removeBondMethod = btClass.getMethod("removeBond");
Boolean returnValue = (Boolean) removeBondMethod.invoke(btDevice);
return returnValue.booleanValue();
}

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

本文程序运行效果如下:

main.xml源码如下:

















工具类ClsUtils.java源码如下:

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的源码如下:

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提高第十三篇之探秘蓝牙隐藏API

Android提高第十三篇之探秘蓝牙隐藏API:目前有70 条留言

  1. 0楼
    kf156:

    [e01]必须的

    2010-11-29 10:24 [回复]
  2. 0楼
    yangc_83:

    [e08]牛人啊~~

    2010-11-29 10:26 [回复]
  3. 被逼来顶,不过不错

    2010-11-29 10:27 [回复]
  4. 0楼
    hellogv:

    回复 qwe285735942:
    ……………………………

    2010-11-29 10:30 [回复]
  5. 0楼
    hmc1985:

    连这个都行,不顶不行呀![e01]

    2010-11-29 10:54 [回复]
  6. 0楼
    pwei007:

    持续跟踪……

    2010-11-29 11:12 [回复]
  7. 0楼
    pwei007:

    [e03]

    2010-11-29 11:12 [回复]
  8. 继续跟进。。。。

    2010-11-29 13:36 [回复]
  9. 0楼
    emike:

    [e01][e01]

    2010-11-29 18:31 [回复]
  10. 0楼
    pwei007:

    哎,可惜做不了测试,没有安卓的手机 [e06]

    2010-11-29 22:03 [回复]
  11. 偶像啊[e03]

    2010-11-30 09:44 [回复]
  12. [e01]

    2010-11-30 11:02 [回复]
  13. 0楼
    surffen:

    [e01]

    2010-11-30 13:30 [回复]
  14. 0楼
    ggggnuirgw:

    动画时间太快了,以后的可能慢点不[e06]

    2010-11-30 14:08 [回复]
  15. [e01]

    2010-11-30 15:19 [回复]
  16. 0楼
    herryz:

    [e01]羡慕。。。

    2010-12-03 17:47 [回复]
  17. 0楼
    moonvos:

    现在正在学嵌入式开发应用程序,想学习下这方面的技术,请高手指点下!

    2010-12-04 12:58 [回复]
  18. 呵呵,我是把源码整个拉下来,找到这个配对函数的。[e01]

    2010-12-09 16:25 [回复]
  19. 0楼
    yanghuaixi:

    [e01]

    2010-12-29 14:23 [回复]
  20. 0楼
    duke136:

    [e01]高人啊!

    2011-01-06 16:35 [回复]
  21. 0楼
    spiderppp:

    确实不错!有个问题文下楼主Settings/src/com/android/settings/bluetooth/CachedBluetoothDevice.java
    里面有个 private final BluetoothDevice mDevice;
    if (!mDevice.createBond()) {
    mLocalManager.showError(mDevice, R.string.bluetooth_error_title,
    R.string.bluetooth_pairing_error_message);
    return;
    }
    BluetoothDevice 的源码那里下。。我下载的platform/packages/apps/Bluetooth.git里面没有,还请楼主指教!

    2011-01-11 15:55 [回复]
  22. 0楼
    hellogv:

    回复 spiderppp:
    base.git

    2011-01-11 16:07 [回复]
  23. 0楼
    mlianghua:

    楼主好强,我觉得我可以试一试,通过反射来直接调硬件编码器,还请楼主指教[e01]

    2011-01-12 09:16 [回复]
  24. 0楼
    goodlinux:

    这次没有被逼顶!!

    2011-01-12 21:25 [回复]
  25. 0楼
    shenmou:

    强帖, 顶

    2011-01-14 10:18 [回复]
  26. 0楼
    yanghuaixi:

    [e01][e01][e01]

    2011-02-21 16:42 [回复]
  27. [e01][e01][e01][e01][e01]

    2011-03-02 21:53 [回复]
  28. [e01]

    2011-03-07 09:58 [回复]
  29. 0楼
    why028328:

    太nx了。。。。我喜欢。。。。

    2011-04-01 10:19 [回复]
  30. 我系中山噶,目前在珠海搞android,有空多多向楼主学习QQ:191972323

    2011-04-12 12:51 [回复]
  31. 楼主,程序非常不错,简洁,而且没有用到ListActivity这个鸡“类”,感觉蛮好的!顶一下~对我的帮助非常大,但是我发现你的代码里面有bug,按那个show按钮的话会强行退出程序的!求解~~谢谢楼主!

    2011-04-15 06:16 [回复]
  32. 0楼
    cshoney:

    我要好好向楼主学习

    2011-04-19 10:20 [回复]
  33. 楼主你好,怎么http://www.pudn.com/downloads305/sourcecode/comm/android/detail1359043.html里面的源码下载不了的呢,麻烦发份到本人邮箱,343827585@qq.com,急需,谢谢。

    2011-04-21 10:40 [回复]
  34. 0楼
    L_e_o_n:

    请问Android蓝牙可以实现不进行自动配对么?
    原因是我要利用串口连接一个没有操作系统的蓝牙发射器,当我手机搜索到这个发射器之后,手机提示我要进行配对操作,但是我发射器没有操作系统,是不能配对的,请问您有什么方法能够避免这个配对的过程么?[e03]

    2011-05-08 00:40 [回复]
  35. 0楼
    hellogv:

    回复 L_e_o_n:
    你的串口蓝牙模块也得有配对码吧,让用户手动输入吧

    2011-05-08 08:40 [回复]
  36. 0楼
    L_e_o_n:

    回复 hellogv:好的,那我再查一下~谢谢![e03]

    2011-05-08 17:08 [回复]
  37. 楼主能否获取到蓝牙的信号强度值啊?

    2011-06-07 13:51 [回复]
  38. 0楼
    hellogv:

    回复 sunrongrong:
    有api,看看文档就知道了

    2011-06-07 19:33 [回复]
  39. 0楼
    sos__sos:

    牛X

    2011-06-21 13:31 [回复]
  40. 楼主,我有个问题,怎样判断远端的bluetoothdevice是手机还是pc??我找了很多方法都没有比较完善的,我想到的就是根据设备上面的服务来判断了

    2011-07-07 20:35 [回复]
  41. 0楼
    hellogv:

    [reply]lighthearts[/reply]
    这些都不重要,关键是能提供什么服务

    2011-07-08 11:58 [回复]
  42. [reply]hellogv[/reply]
    其实是这样的,我在做一个搜索设备的界面,搜索到的设备是手机的话,就要贴手机的图,是电脑的话就要贴电脑的图,所以需要知道device是电脑还是手机,现在还在纠结中

    2011-07-08 12:36 [回复]
  43. 0楼
    hellogv:

    [reply]lighthearts[/reply]
    这个还挺难的。。。。。。。。

    2011-07-08 16:09 [回复]
  44. 0楼
    haojunming:

    大哥,我现在在做蓝牙的一对多连接,如果我要是手机作为服务器,同时接收几个蓝牙设备,那么这个uuid该怎么配置,谢谢啊

    2011-08-01 10:59 [回复]
  45. 0楼
    hellogv:

    [reply]haojunming[/reply]
    蓝牙本身就支持1对7的链接

    2011-08-01 11:31 [回复]
  46. 0楼
    haojunming:

    可是我的那个是串口连接,需要对应的uuid,我不知道该怎么设置,您能不能给提供几个对应的uuid,我自己从网上找了两个,但是还需要五个,蓝牙串口服务 SerialPortServiceClass_UUID: TGUID = '{00001101-0000-1000-8000-00805F9B34FB}';
    LANAccessUsingPPPServiceClass_UUID: TGUID = '{00001102-0000-1000-8000-00805F9B34FB}';

    2011-08-01 11:47 [回复]
  47. 0楼
    haojunming:

    :蓝牙本身就支持1对7的链接但是我需要有uuid才可以, 现在我是找不到那么多的uuid,您知道它是怎么取得的吗?或者说它是怎么生成的,有相应的代码吗?能不能给发一份,谢谢啊,我的手机号18701427983,qq1149598411

    2011-08-01 11:49 [回复]
  48. 0楼
    hellogv:

    [reply]haojunming[/reply]
    。。。。。。。。。。。。。。。一个SPP就已经可以1对7了。。。。。。。。。。。。。。

    2011-08-03 19:11 [回复]
  49. String[] values = str.split("//|");  
                String address=values[2]; 
    应该为:
    String[] values = str.split("|");  
                String address=values[1]; 
    数组下标从0开始的

    2011-08-10 11:14 [回复]
  50. 0楼
    haojunming:

    大哥,问一下,android蓝牙自动配对该怎么办啊,我自己从网上找了好多这样的东西可是都不合适,能不能给写一个可用的实例代码,谢谢啊

    2011-08-12 10:03 [回复]
  51. 0楼
    muyiangel7:

    最近在做android蓝牙这方面,想请问楼主怎样才能消除掉蓝牙匹配的弹出框(就是不用用户点击“是”这个按钮就能完成匹配,对用户要隐藏)?跪求指导!

    2011-08-15 09:28 [回复]
  52. 0楼
    haojunming:

    谁知道android蓝牙自动配对怎么实现,谢谢啊,我一直在找这个的实现代码,但是很不幸,到现在找了一个星期 了,依然没有解决方案,我的qq1149598411

    2011-08-15 11:40 [回复]
  53. 0楼
    hellogv:

    [reply]haojunming[/reply]
    我也尝试过自动配对,但这个不是Framework层的,更底层来实现

    2011-08-16 12:20 [回复]
  54. 0楼
    yuanbieli:

    [reply]lighthearts[/reply]
    这个android本身应该是已经实现了。
    看settings的源码,android已经能区分搜索到的设备是PC还是手机。
    参考:com.android.settings.bluetooth.CachedBluetoothDevice.java
    getBtClassDrawable()方法中的如下代码,
    switch (mBtClass.getMajorDeviceClass()) {
    case BluetoothClass.Device.Major.COMPUTER:
    return R.drawable.ic_bt_laptop;

    case BluetoothClass.Device.Major.PHONE:
    return R.drawable.ic_bt_cellphone;}
    已经实现根据设备不同选用不同图标。
    核心类用到BluetoothClass.java
    供参考~

    2011-08-19 11:38 [回复]
  55. 0楼
    haojunming:

    [reply]hellogv[/reply]
    我实现了自动配对,可是配对成功后依然会弹出那个配对框, 只是很快就又消失了,我分析是,我自己对那个配对的广播进行了监听,在onReceive方法里面手动输入配对码,可是系统也对这个进行监听,所以都对此作出了处理,这样才造成这个结局,不知道有没有什么好的解决方案。我们公司的项目经理催我很厉害了,帮帮忙。如果真的不能在framework层实现自动配对的话,能不能告诉我为什么啊,谢谢

    2011-08-23 17:42 [回复]
  56. 0楼
    haojunming:

    [reply]hellogv[/reply]
    或者你能不能写一篇文章就是专门关于android蓝牙在framework层不能实现自动配对的文章?

    2011-08-23 18:10 [回复]
  57. 0楼
    haojunming:

    [reply]muyiangel7[/reply]
    我也在做这个蓝牙自动配对,不知道你有什么解决方案吗

    2011-08-24 08:34 [回复]
  58. 0楼
    hellogv:

    [reply]haojunming[/reply]
    因为framework也是直接调用底层蓝牙驱动来实现的,这里framework也是无济于事

    2011-08-24 18:37 [回复]
  59. 0楼
    haojunming:

    [reply]hellogv[/reply]
    问您一件事情,android中 Setting里面的那些设置信息都保存在哪里了?

    2011-08-25 14:22 [回复]
  60. 0楼
    haojunming:

    [reply]hellogv[/reply]
    我的一个设想是android当中的设置配对信息应该是保存起来的,那么我手动的把配对信息写进去行吗?

    2011-08-25 14:24 [回复]
  61. 0楼
    hellogv:

    [reply]haojunming[/reply]
    我也试过,用已知的配对码去尝试自动配对,但是没接口可以调用自动配对的功能啊。

    2011-08-25 22:12 [回复]
  62. 0楼
    haojunming:

    [reply]hellogv[/reply]
    您知道怎么在代码里面实现蓝牙解除配对吗?

    2011-09-14 10:46 [回复]
  63. 0楼
    hellogv:

    [reply]haojunming[/reply]
    你看看framework里有没有函数,我记得貌似有

    2011-09-15 18:58 [回复]
  64. 0楼
    java123yy:

    java123yy@163.com
    写得真不错,目前我正在解决蓝牙这一块的问题,网上深入研究的,貌似就你这一个了。
    所以,楼主,诚心求该蓝牙源码。谢谢啦!

    2011-09-28 16:04 [回复]
  65. 0楼
    hellogv:

    [reply]java123yy[/reply]
    什么蓝牙源码?

    2011-09-28 18:59 [回复]
  66. 0楼
    liubande:

    [reply]edifier_jcc[/reply]
    class ClickEvent implements View.OnClickListener {
    。。。。。。。。
    else if(v==btnShow){
    //这里修改一下,用对象可能造成空指针,用类名就ok了
    ClsUtils.printAllInform(BluetoothDevice.class);
    }

    }

    }

    2011-10-12 10:23 [回复]
  67. 0楼
    ahjxly:

    楼主,跪求蓝牙连接部分的源代码啊 QQ1012546297,谢谢啦

    2012-03-20 14:14 [回复]
  68. 0楼
    ahjxly:

    蓝牙连接时怎么弄的哦?貌似是用createRfcommSocket反射调用呢,还是connect??通过UUID传递是不是不可以??

    2012-03-20 14:18 [回复]
  69. 0楼
    vipclx:

    大神啊~~~在网上搜了半天发现蓝牙讲解有技术含量的仅此一家,我最近也在弄蓝牙方面,因为初次涉及,好多不懂,能加我QQ指点下迷津吗?感觉不尽,我QQ513644511,谢谢

    2012-06-07 19:41 [回复]
  70. 0楼
    hellogv:

    [reply]vipclx[/reply]
    哥我很久不弄蓝牙了。。。。

    2012-06-11 12:59 [回复]

发表评论


QQ群互动

Linux系统与内核学习群:194051772

WP建站技术学习交流群:194062106

魔豆之路QR

魔豆的Linux内核之路

魔豆的Linux内核之路

优秀工程师当看优秀书籍

优秀程序员,要看优秀书!

赞助商广告

友荐云推荐