美文网首页爱码蛋蓝牙Android知识
Android 蓝牙搜索,配对,连接发送数据

Android 蓝牙搜索,配对,连接发送数据

作者: monkeynessss | 来源:发表于2017-03-13 18:18 被阅读5979次

首先需要在清单配置里面添加两个权限:

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

android里面蓝牙是通过BluetoothAdapter来进行操作的,所以首先我们需要获取到BluetoothAdapter的实例

//先获取BlueToothAdapter的实例
BluetoothAdapter blueToothAdapter = BluetoothAdapter.getDefaultAdapter();

在搜索之前,我们可以先获取与我们配对过的设备

//获取已经配对过的设备的集合
Set<BluetoothDevice> bondedDevices = blueToothAdapter.getBondedDevices()

但是想要获取到配对过的设备,我们必须是在

//手机有蓝牙设备并且蓝牙是打开的
blueToothAdapter != null && blueToothAdapter.isEnabled()

下面我们谈一下蓝牙的<b>打开方式</b>,目前作者知道的方式有3种
第一种:

//强制打开蓝牙
blueToothAdapter.enable();

第二种:

//会以dialog的形式打开一个activity,并且如果我们通过startActivityForResult的形式的话
//还能查看蓝牙是否被打开,或者处理蓝牙被打开之后的操作
//如果是result_ok的话那么是打开,反之打开失败
startActivityForResult(new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE));

第三种:

//设置本地设备可以被其它设备搜索,可被搜索的时间是有限的,最多为300s
//效果和第二种类似
Intent discoverableIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_DISCOVERABLE);
discoverableIntent.putExtra(BluetoothAdapter.EXTRA_DISCOVERABLE_DURATION, 300);
startActivity(discoverableIntent);

<b>关闭蓝牙</b>我们可以调用

blueToothAdapter.disable();

接下来是<b>蓝牙搜索</b>
首先如果想要开启蓝牙搜索,那么只需要调用

blueToothAdapter.startDiscovery();

但是我们该如何接收我们搜索到的设备呢,很明显当然是通过接收广播的形式来接收。所以我们应该自定义一个广播接收器,

class BluetoothReceiver extends BroadcastReceiver {

        @Override
        public void onReceive(Context context, Intent intent) {

            if (intent.getAction().equals(BluetoothDevice.ACTION_FOUND)) {
                Log.e(getPackageName(), "找到新设备了");
                BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
            }
        }
    }

如上代码,我们可以获取到搜索到的蓝牙设备。我们只需在代码里面注册这个广播接收器,在调用蓝牙的搜索方法,就能够进行蓝牙的搜索了。

//注册广播
IntentFilter intentFilter = new IntentFilter();
intentFilter.addAction(BluetoothDevice.ACTION_FOUND);
registerReceiver(new BluetoothReceiver(), intentFilter);

这里需要注意的是,如果你的代码将运行在(Build.VERSION.SDK_INT >= 23)的设备上,那么务必加上以下权限,并在代码中动态的申请权限

<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/>
private void requestPermission() {
        if (Build.VERSION.SDK_INT >= 23) {
            int checkAccessFinePermission = ActivityCompat.checkSelfPermission(this, Manifest.permission.ACCESS_FINE_LOCATION);
            if (checkAccessFinePermission != PackageManager.PERMISSION_GRANTED) {
                ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.ACCESS_FINE_LOCATION},
                        REQUEST_PERMISSION_ACCESS_LOCATION);
                Log.e(getPackageName(), "没有权限,请求权限");
                return;
            }
            Log.e(getPackageName(), "已有定位权限");
            //这里可以开始搜索操作
        }
    }

    @Override
    public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) {
        switch (requestCode) {
            case REQUEST_PERMISSION_ACCESS_LOCATION: {
                if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                    Log.e(getPackageName(), "开启权限permission granted!");
                    //这里可以开始搜索操作
                } else {
                    Log.e(getPackageName(), "没有定位权限,请先开启!");
                }
            }
        }
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    }

再来就是<b>蓝牙的配对</b>,目前楼主采用的是通过反射的形式来调用BluetoothDevice的createBondde()方法,如果我们想监听配对的这个过程,那么我们可以为广播接收器再注册一个action。

try {
      //如果想要取消已经配对的设备,只需要将creatBond改为removeBond
       Method method = BluetoothDevice.class.getMethod("createBond");
       Log.e(getPackageName(), "开始配对");
       method.invoke(device);
    } catch (Exception e) {
      e.printStackTrace();
    }
//绑定状态发生变化
intentFilter.addAction(BluetoothDevice.ACTION_BOND_STATE_CHANGED);

广播接收器里面我们可以这样写

if (intent.getAction().equals(BluetoothDevice.ACTION_BOND_STATE_CHANGED)) {
                BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
                switch (device.getBondState()) {
                    case BluetoothDevice.BOND_NONE:
                        Log.e(getPackageName(), "取消配对");
                        break;
                    case BluetoothDevice.BOND_BONDING:
                        Log.e(getPackageName(), "配对中");
                        break;
                    case BluetoothDevice.BOND_BONDED:
                        Log.e(getPackageName(), "配对成功");
                        break;
                }
}

再来就是<b>向已经配对的设备发送数据</b>
发送数据分为服务端和客户端,通过socket来进行消息的交互。

服务端

new Thread(new Runnable() {
            @Override
            public void run() {
                InputStream is = null;
                try {
                    BluetoothServerSocket serverSocket = blueToothAdapter.listenUsingRfcommWithServiceRecord("serverSocket", uuid);
                    mHandler.sendEmptyMessage(startService);
                    BluetoothSocket accept = serverSocket.accept();
                    is = accept.getInputStream();

                    byte[] bytes = new byte[1024];
                    int length = is.read(bytes);

                    Message msg = new Message();
                    msg.what = getMessageOk;
                    msg.obj = new String(bytes, 0, length);
                    mHandler.sendMessage(msg);
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }).start();

客户端

new Thread(new Runnable() {
            @Override
            public void run() {
                OutputStream os = null;
                try {
                    BluetoothSocket socket = strArr.get(i).createRfcommSocketToServiceRecord(uuid);
                    socket.connect();
                    os = socket.getOutputStream();
                    os.write("testMessage".getBytes());
                    os.flush();
                    mHandler.sendEmptyMessage(sendOver);
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }).start();

可以看到无论是服务端还是客户端,都需要新起一个子线程来操作。那么服务端和客户端是怎么识别对方的呢,那么就需要用到<b>UUID</b>了,只有当服务端和客户端的<b>UUID</b>相同的时候才能够建立连接。

<b>蓝牙连接发送数据的时候的坑</b>
当博主第一次写好客户端服务端测试的时候,服务端一直报错

java.io.IOException: bt socket closed, read return: -1

也就是这句话报错

is.read(bytes)

当时一直以为是读取返回值是-1就会报错,但是不对啊,以前这么写也没错过,在网上百度了半天,看了别人的博客论坛也没有解决办法,最后才注意到报错的前一句,

bt socket closed

之前一直认为是什么服务端这边,报错连接才会断开,后来一想客户端也会断开连接啊,这才找到问题所在。原来是我当时write完数据之后将连接关闭了,这才导致服务端这边连接断开的。所以重要的事情要说3遍,千万别急着断开连接,千万别急着断开连接,千万别急着断开连接。

<a href="https://github.com/Monkeynessss/BlueToothDemo">github源代码链接</a>

相关文章

  • Android 蓝牙搜索,配对,连接发送数据

    首先需要在清单配置里面添加两个权限: android里面蓝牙是通过BluetoothAdapter来进行操作的,所...

  • android bluetooth——蓝牙的开启、搜索、配对与连

    android bluetooth——蓝牙的开启、搜索、配对与连接 在连接一次后退出再连接蓝牙设备时会报以下错误:...

  • Android 蓝牙设备通讯

    Android 蓝牙设备通讯的开发(配对/连接/传输数据)http://blog.csdn.net/qq_3055...

  • Android 蓝牙开发实践笔记

    本文基于传统蓝牙开发。 先来梳理一下蓝牙开发的逻辑,基本分为以下几步:搜索设备,配对设备,连接设备,传输数据。前三...

  • 蓝牙连接

    1.蓝牙搜索 2.蓝牙连接 3.注册收发数据的通知 **##4.发送数据 5.接收数据 下载Demo--> 我的g...

  • 蓝牙ble

    绑定、配对、连接蓝牙4.0BLE介绍_casuallystone_新浪博客 Android_低功耗 Android...

  • Android BLE蓝牙详细解读(一)

    本文主要讲解Android低功耗蓝牙的api使用以及蓝牙扫描、连接、发送数据、接收数据等一系列操作,本篇结尾有本人...

  • Bluetooth和苹果MFi认证相关总结

    背景:公司开发项目中需要手机app连接配对手表类的蓝牙设备,android设备很容易就跟蓝牙设备连接了,但是苹果手...

  • Android之Bluetooth配对

    前言 解读JniCallbacks 场景:配对建立连接过程 案例:蓝牙配对连接注:配对的前提,设备已知,代码分析要...

  • Android经典蓝牙通讯传输DEMO

    BlueUtils 项目地址:GitHub 经典蓝牙搜索,连接,数据传输小DEMO 通过经典模式 搜索 蓝牙应用。...

网友评论

  • 5cf8848faad5:我这种拿来主义,这几天试了好多蓝牙的demo都连接不上,总是在socket.connect那里连接不上。楼主的这个拿两个设备一个开服务接收一个发送才能收到,所以想问一下,手机那种的蓝牙接收是怎么搞的啊?
    monkeynessss:@summer714 你们要做的是什么功能 手机连接硬件设备?
  • 2b1377778ded:运行到socket.connect();就会报错-->java.io.IOException: read failed, socket might closed or timeout, read ret: -1。
    我并没有任何关闭连接的代码。前面匹配都是ok的,这个问题困扰我两天了,求帮助
    2b1377778ded:case BluetoothDevice.BOND_BONDED:
    //配对成功
    直到配对成功我就开始调用客户端线程,直接cope你写的,socket.connect();执行就开始报错。
    我用的androud 7.0,开发的api是26,大神求帮助:pray:
    2b1377778ded:@monkeynessss 我只在sendMessage()这个方法中调用了socket.connect();就报错了,我还没有开始发送数据,整个类中我也并没有手动调用过关闭socket。不过我当时还没有调用getMessage()方法,也就是说我只开启了发送的线程,会有问题么?
    monkeynessss:@王飞_f5f2 你看看是不是客户端写完数据关掉了流
  • 米奇小林:大神你好,那个 反射的 配对方法 好像在 6.0手机 没有反应。
    敲代码的鸡:现在搞得怎么样了
    米奇小林:@monkeynessss 是我自己弄错了,我把自定义的device传进去了。:smile:
    monkeynessss:@米奇小林 不会啊朋友,我当时用的就是6.0的手机,你手机哪个厂家的
  • NecronsKnight:谢谢大佬,我最近在学习通过蓝牙手机互传数据,还有通过手机和CC2541互传数据,不是太会很头疼,读了您的文章我有了点想法,谢谢
  • ec5b10c9f70b:这个源码不是应该有客户端和服务端两个apk吗,但是你的只有一个是怎么回事呢
  • e206c3b457f5:请问楼主,有的设备的不需要配对,那个同信是怎么实现的呢?我用你的方法不行,这个好像只使用于要配对的设备;比如与音响连接等
    e206c3b457f5:好像不是的,有一种可以通过GATTService来获取服务和特征的,它可以直接建立连接,可跳过配对,但是我搞不来,好蛋疼!
    monkeynessss: @dengfuyao 蓝牙通信都需要配对才能连接的
  • 枫雪狼情:里面的两个重点 权限 和 uuid 没注意的话会走很多弯路

本文标题:Android 蓝牙搜索,配对,连接发送数据

本文链接:https://www.haomeiwen.com/subject/tejfnttx.html