Android蓝牙操作笔记

蓝牙是一种支持设备短距离传输数据的无线技术。android在2.0以后提供了这方面的支持。 
从查找蓝牙设备到能够相互通信要经过几个基本步骤(本机做为服务器): 
1.设置权限 
在manifest中配置

  1. <uses-permission android:name="android.permission.BLUETOOTH"/>
  2. <uses-permission android:name="android.permission.BLUETOOTH_ADMIN"/>

2.启动蓝牙 
首先要查看本机是否支持蓝牙,获取BluetoothAdapter蓝牙适配器对象

  1. BluetoothAdapter mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
  2. if(mBluetoothAdapter == null){
  3. //表明此手机不支持蓝牙
  4. return;
  5. }
  6. if(!mBluetoothAdapter.isEnabled()){ //蓝牙未开启,则开启蓝牙
  7. Intent enableIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE);
  8. startActivityForResult(enableIntent, REQUEST_ENABLE_BT);
  9. }
  10. //......
  11. public void onActivityResult(int requestCode, int resultCode, Intent data){
  12. if(requestCode == REQUEST_ENABLE_BT){
  13. if(requestCode == RESULT_OK){
  14. //蓝牙已经开启
  15. }
  16. }
  17. }

3。发现蓝牙设备 
这里可以细分为几个方面 
(1)使本机蓝牙处于可见(即处于易被搜索到状态),便于其他设备发现本机蓝牙

  1. //使本机蓝牙在300秒内可被搜索
  2. private void ensureDiscoverable() {
  3. if (mBluetoothAdapter.getScanMode() !=
  4. BluetoothAdapter.SCAN_MODE_CONNECTABLE_DISCOVERABLE) {
  5. Intent discoverableIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_DISCOVERABLE);
  6. discoverableIntent.putExtra(BluetoothAdapter.EXTRA_DISCOVERABLE_DURATION, 300);
  7. startActivity(discoverableIntent);
  8. }
  9. }

(2)查找已经配对的蓝牙设备,即以前已经配对过的设备

  1. Set<BluetoothDevice> pairedDevices = mBluetoothAdapter.getBondedDevices();
  2. if (pairedDevices.size() > 0) {
  3. findViewById(R.id.title_paired_devices).setVisibility(View.VISIBLE);
  4. for (BluetoothDevice device : pairedDevices) {
  5. //device.getName() +" "+ device.getAddress());
  6. }
  7. } else {
  8. mPairedDevicesArrayAdapter.add("没有找到已匹对的设备");
  9. }

(3)通过mBluetoothAdapter.startDiscovery();搜索设备,要获得此搜索的结果需要注册 
一个BroadcastReceiver来获取。先注册再获取信息,然后处理

  1. //注册,当一个设备被发现时调用onReceive
  2. IntentFilter filter = new IntentFilter(BluetoothDevice.ACTION_FOUND);
  3. this.registerReceiver(mReceiver, filter);
  4. //当搜索结束后调用onReceive
  5. filter = new IntentFilter(BluetoothAdapter.ACTION_DISCOVERY_FINISHED);
  6. this.registerReceiver(mReceiver, filter);
  7. //.......
  8. private BroadcastReceiver mReceiver = new BroadcastReceiver() {
  9. @Override
  10. public void onReceive(Context context, Intent intent) {
  11. String action = intent.getAction();
  12. if(BluetoothDevice.ACTION_FOUND.equals(action)){
  13. BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
  14. // 已经配对的则跳过
  15. if (device.getBondState() != BluetoothDevice.BOND_BONDED) {
  16. mNewDevicesArrayAdapter.add(device.getName() + "\n" + device.getAddress());  //保存设备地址与名字
  17. }
  18. }else if (BluetoothAdapter.ACTION_DISCOVERY_FINISHED.equals(action)) {  //搜索结束
  19. if (mNewDevicesArrayAdapter.getCount() == 0) {
  20. mNewDevicesArrayAdapter.add("没有搜索到设备");
  21. }
  22. }
  23. }
  24. };

4.建立连接 
查找到设备 后,则需要建立本机与其他设备之间的连接。 
一般用本机搜索其他蓝牙设备时,本机可以作为一个服务端,接收其他设备的连接。 
启动一个服务器端的线程,死循环等待客户端的连接,这与ServerSocket极为相似。 
这个线程在准备连接之前启动

  1. //UUID可以看做一个端口号
  2. private static final UUID MY_UUID =
  3. UUID.fromString("fa87c0d0-afac-11de-8a39-0800200c9a66");
  4. //像一个服务器一样时刻监听是否有连接建立
  5. private class AcceptThread extends Thread{
  6. private BluetoothServerSocket serverSocket;
  7. public AcceptThread(boolean secure){
  8. BluetoothServerSocket temp = null;
  9. try {
  10. temp = mBluetoothAdapter.listenUsingRfcommWithServiceRecord(
  11. NAME_INSECURE, MY_UUID);
  12. } catch (IOException e) {
  13. Log.e("app", "listen() failed", e);
  14. }
  15. serverSocket = temp;
  16. }
  17. public void run(){
  18. BluetoothSocket socket=null;
  19. while(true){
  20. try {
  21. socket = serverSocket.accept();
  22. } catch (IOException e) {
  23. Log.e("app", "accept() failed", e);
  24. break;
  25. }
  26. }
  27. if(socket!=null){
  28. //此时可以新建一个数据交换线程,把此socket传进去
  29. }
  30. }
  31. //取消监听
  32. public void cancel(){
  33. try {
  34. serverSocket.close();
  35. } catch (IOException e) {
  36. Log.e("app", "Socket Type" + socketType + "close() of server failed", e);
  37. }
  38. }
  39. }

搜索到设备后可以获取设备的地址,通过此地址获取一个BluetoothDeviced对象,可以看做客户端,通过此对象device.createRfcommSocketToServiceRecord(MY_UUID);同一个UUID可与服务器建立连接获取另一个socket对象,由此服务端与客户端各有一个socket对象,此时 
他们可以互相交换数据了。 
创立客户端socket可建立线程

  1. //另一个设备去连接本机,相当于客户端
  2. private class ConnectThread extends Thread{
  3. private BluetoothSocket socket;
  4. private BluetoothDevice device;
  5. public ConnectThread(BluetoothDevice device,boolean secure){
  6. this.device = device;
  7. BluetoothSocket tmp = null;
  8. try {
  9. tmp = device.createRfcommSocketToServiceRecord(MY_UUID_SECURE);
  10. } catch (IOException e) {
  11. Log.e("app", "create() failed", e);
  12. }
  13. }
  14. public void run(){
  15. mBluetoothAdapter.cancelDiscovery();    //取消设备查找
  16. try {
  17. socket.connect();
  18. } catch (IOException e) {
  19. try {
  20. socket.close();
  21. } catch (IOException e1) {
  22. Log.e("app", "unable to close() "+
  23. " socket during connection failure", e1);
  24. }
  25. connetionFailed();  //连接失败
  26. return;
  27. }
  28. //此时可以新建一个数据交换线程,把此socket传进去
  29. }
  30. public void cancel() {
  31. try {
  32. socket.close();
  33. } catch (IOException e) {
  34. Log.e("app", "close() of connect  socket failed", e);
  35. }
  36. }
  37. }

5.建立数据通信线程,进行读取数据

    1. //建立连接后,进行数据通信的线程
    2. private class ConnectedThread extends Thread{
    3. private BluetoothSocket socket;
    4. private InputStream inStream;
    5. private OutputStream outStream;
    6. public ConnectedThread(BluetoothSocket socket){
    7. this.socket = socket;
    8. try {
    9. //获得输入输出流
    10. inStream = socket.getInputStream();
    11. outStream = socket.getOutputStream();
    12. } catch (IOException e) {
    13. Log.e("app", "temp sockets not created", e);
    14. }
    15. }
    16. public void run(){
    17. byte[] buff = new byte[1024];
    18. int len=0;
    19. //读数据需不断监听,写不需要
    20. while(true){
    21. try {
    22. len = inStream.read(buff);
    23. //把读取到的数据发送给UI进行显示
    24. Message msg = handler.obtainMessage(BluetoothChat.MESSAGE_READ,
    25. len, -1, buff);
    26. msg.sendToTarget();
    27. } catch (IOException e) {
    28. Log.e("app", "disconnected", e);
    29. connectionLost();   //失去连接
    30. start();    //重新启动服务器
    31. break;
    32. }
    33. }
    34. }
    35. public void write(byte[] buffer) {
    36. try {
    37. outStream.write(buffer);
    38. // Share the sent message back to the UI Activity
    39. handler.obtainMessage(BluetoothChat.MESSAGE_WRITE, -1, -1, buffer)
    40. .sendToTarget();
    41. } catch (IOException e) {
    42. Log.e("app", "Exception during write", e);
    43. }
    44. }
    45. public void cancel() {
    46. try {
    47. socket.close();
    48. } catch (IOException e) {
    49. Log.e("app", "close() of connect socket failed", e);
    50. }
    51. }
    52. }
时间: 2024-11-14 22:16:09

Android蓝牙操作笔记的相关文章

Android蓝牙操作笔记(转)

蓝牙是一种支持设备短距离传输数据的无线技术.android在2.0以后提供了这方面的支持. 从查找蓝牙设备到能够相互通信要经过几个基本步骤(本机做为服务器): 1.设置权限 在manifest中配置 <uses-permission android:name="android.permission.BLUETOOTH"/> <uses-permission android:name="android.permission.BLUETOOTH_ADMIN&qu

Android 蓝牙操作详解

1.启用蓝牙并使设备处于可发现状态 1.1 在使用BluetoothAdapter类的实例进操作之前,应启用isEnable()方法检查设备是否启用了蓝牙适配器. // 使用意图提示用户启用蓝牙,并使设备处于可发现状态 private void startBluetooth() { BluetoothAdapter bt = BluetoothAdapter.getDefaultAdapter(); // 检测蓝牙是否开启 if (!bt.isEnabled()) { Intent enable

Android 蓝牙学习笔记(一)

首先扯点别的:这是我自己看蓝牙方面的知识的笔记,要是大家想看蓝牙方面的知识,可以看看链接中的文章,我看了看,写的也是没谁了,完全是android官网的翻译加详解,非常完美!http://blog.csdn.net/small_lee/article/details/50800722 估计一篇文章也写不完,应该写两篇,先给大家上图 第一:这篇文章完成的任务 现在就按照图片按钮的顺序从头往下讲解.完整的代码在文章的末尾贴上.使用蓝牙方面的功能需要在Androidmanifest.xml文件中添加两个

Android蓝牙串口通信模板

转载请注明出处,谢谢http://blog.csdn.net/metalseed/article/details/7988945 Android蓝牙操作:与蓝牙串口模块通信,或其他蓝牙设备通信. 初涉android的蓝牙操作,按照固定MAC地址连接获取Device时,程序始终是异常终止,查了好多天代码都没查出原因.今天改了一下API版本,突然就成功连接了.总结之后发现果然是个坑爹之极的错误. 为了这种错误拼命查原因浪费大把时间是非常不值得的,但是问题不解决更是揪心.可惜我百度了那么多,都没有给出

Android:日常学习笔记(10)———使用LitePal操作数据库

Android:日常学习笔记(10)---使用LitePal操作数据库 引入LitePal 什么是LitePal LitePal是一款开源的Android数据库框架,采用了对象关系映射(ORM)的模式,将平时开发时最常用的一些数据库功能进行了封装,使得开发者不用编写一行SQL语句就可以完成各种建表.増删改查的操作.并且LitePal很"轻",jar包大小不到100k,而且近乎零配置,这一点和Hibernate这类的框架有很大区别.目前LitePal的源码已经托管到了GitHub上. 关

Android 蓝牙的常用操作

最近对Android设备的蓝牙操作进行了一些研究, 下面做一些总结, 版本是4.4,列出的解决方案多来源于网络,感谢强大的网友们: 操作蓝牙可以分为常规的操作,和非常规的操作.所谓常规的操作,就是界面上有提示,需要客户许可进行的一些操作.非常规的则通常是采用反射等手段,达到不知不觉连接蓝牙的目的. 一. 常规操作: 1. 获取蓝牙的操作接口: BluetoothAdapter mBtAdapter = BluetoothAdapter.getDefaultAdapter(); 蓝牙的相关操作基本

Android学习之蓝牙操作

Android 蓝牙是一种支持设备短距离传输数据的无线技术.android在2.0以后提供了这方面的支持. 从查找蓝牙设备到能够相互通信要经过几个基本步骤(本机做为服务器): 1.设置权限 在manifest中配置 Xml代码   <uses-permission android:name="android.permission.BLUETOOTH"/> <uses-permission android:name="android.permission.BL

java/android 设计模式学习笔记(14)---外观模式

这篇博客来介绍外观模式(Facade Pattern),外观模式也称为门面模式,它在开发过程中运用频率非常高,尤其是第三方 SDK 基本很大概率都会使用外观模式.通过一个外观类使得整个子系统只有一个统一的高层的接口,这样能够降低用户的使用成本,也对用户屏蔽了很多实现细节.当然,在我们的开发过程中,外观模式也是我们封装 API 的常用手段,例如网络模块.ImageLoader 模块等.其实我们在开发过程中可能已经使用过很多次外观模式,只是没有从理论层面去了解它. 转载请注明出处:http://bl

java/android 设计模式学习笔记(一)---单例模式

前段时间公司一些同事在讨论单例模式(我是最渣的一个,都插不上嘴 T__T ),这个模式使用的频率很高,也可能是很多人最熟悉的设计模式,当然单例模式也算是最简单的设计模式之一吧,简单归简单,但是在实际使用的时候也会有一些坑. PS:对技术感兴趣的同鞋加群544645972一起交流 设计模式总目录 java/android 设计模式学习笔记目录 特点 确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例. 单例模式的使用很广泛,比如:线程池(threadpool).缓存(cache).对