Android中BLE连接出现“BluetoothGatt status 133”的解决方法

2023-11-15

http://www.loverobots.cn/android-ble-connection-solution-bluetoothgatt-status-133.html

前  言

最近的工作方向一直在低功耗蓝牙方面,也就是BLE(Bluetoooth Low Energy)。要说起蓝牙,之前能够想到的应用也就是蓝牙耳机、蓝牙手柄之类的。本以为蓝牙这个技术会慢慢没落下去,不过,现在随着智能设备的流行,出现了智能手环这样的可穿戴设备,而其实现通信的方式就是蓝牙。

起初,JACK的机器人怀着做整套APP的心态,觉得蓝牙模块无非是了解一下协议,模仿一下类似实现方式就可以了。直到折腾了一个月后才发现,BLE真TM是个大坑(额,爆粗口了,大家请见谅...),不仅涉及到应用层,还涉及到底层协议栈,需要设置从设备(Peripheral)的话还得会单片机、C语言的混合编程。而最坑的一点是,它还不稳定——不稳定还开发个毛线啊!!!所以,开发的重点之一也包括,解决稳定性问题......

这篇文章先重点记录一下遇到的这个“BluetoothGatt status 133”问题,后续会把相关经验慢慢整理出来分享给大家,也为自己将来再次遇到蓝牙开发时做个知识储备。

BluetoothGatt status 133

首先简单给大家解释下什么是BLE中的“BluetoothGatt status 133”。

在BLE通信中,或者说整个蓝牙通信中,都会经过以下基本步骤:

  • 发现设备->配对/绑定设备->建立连接->数据通信

status 133这个错误状态就出现在BLE通信步骤中的“建立连接”过程中,而这个错误也是网络上进行BLE开发基本都会遇到的一个错误。但是,因为BLE开发还属于比较前沿的技术(公司大牛说的,感觉也还算吧),可供查阅参考的资料大部分还是来自国外的英文资料,国内可供参考的实在是少之又少。

如果有小伙伴在百度上搜索一下BLE相关问题,经常会看到有一些人在提问,然后下面一大片全都是统一回复——“我也遇到了这个问题,楼主解决了请分享一下......”。

而JACK的机器人遇到此问题的情况是,在与一台从设备建立连接后,退出测试DEMO,同时调用了各种释放资源的方法后,再次进入DEMO,与从设备建立连接会出现无法成功建立连接的问题,只要重复多次进入就又可以成功。

而根据stackoverflow上的相关提问:

Android中BLE连接出现“BluetoothGatt status 133”的解决方法

可以发现解决此错误的重点是“在再次和从设备建立连接时,需要首先调用BluetoothGatt的close()释放掉之前的连接资源”,而原因是对于主设备来说,与从设备的连接数量是有限制的,比如其中提到的针对Galaxy s4手机的6个连接数量限制。

那么,重点还是回到连接资源的释放问题。下面看看JACK的机器人的解决方法,代码片段如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
private final BluetoothGattCallback mGattCallback = new BluetoothGattCallback() {
     @Override
     public void onConnectionStateChange(BluetoothGatt gatt, int status, int newState) {
         String intentAction;
         if (status == BluetoothGatt.GATT_SUCCESS) {
             if (newState == BluetoothProfile.STATE_CONNECTED) {
                 intentAction = BluetoothConstants.ACTION_GATT_CONNECTED;
                 mBLEConnectionState = BluetoothConstants.BLE_STATE_CONNECTED;
                 broadcastUpdate(intentAction);
                 Log.i(TAG, "Connected to GATT server." );
                 Log.i(TAG, "Attempting to start service discovery:" + mBluetoothGatt.discoverServices());
             } else if (newState == BluetoothProfile.STATE_DISCONNECTED) {
                 intentAction = BluetoothConstants.ACTION_GATT_DISCONNECTED;
                 mBLEConnectionState = BluetoothConstants.BLE_STATE_DISCONNECTED;
                 close(); // 防止出现status 133
                 Log.i(TAG, "Disconnected from GATT server." );
                 broadcastUpdate(intentAction);
             }
         } else {
             Log.d(TAG, "onConnectionStateChange received: " + status);
             intentAction = BluetoothConstants.GATT_STATUS_133;
             mBLEConnectionState = BluetoothConstants.BLE_STATE_DISCONNECTED;
             close(); // 防止出现status 133
             broadcastUpdate(intentAction);
             connect(reConnectAddress);
         }
     }
 
     @Override
     public void onServicesDiscovered(BluetoothGatt gatt, int status) {
         if (status == BluetoothGatt.GATT_SUCCESS) {
             Log.d(TAG, "mBluetoothGatt = " + mBluetoothGatt);
             broadcastUpdate(BluetoothConstants.ACTION_GATT_SERVICES_DISCOVERED);
             BluetoothGattService gattService = mBluetoothGatt
                     .getService(UUID.fromString(UUIDS.KARINGUPGRADE_SERVICE_UUID));
             if (gattService == null ) {
                 showMessage( "gattService not found!" );
                 broadcastUpdate(BluetoothConstants.DEVICE_DOES_NOT_SUPPORT_UART);
                 return ;
             }
             Log.d(TAG, "onServicesDiscovered************************gattService=" + gattService);
             BluetoothGattCharacteristic character = gattService
                     .getCharacteristic(UUID.fromString(UUIDS.RX_CHAR_UUID));
             enableRXNotification( true , character);
         } else {
             Log.d(TAG, "onServicesDiscovered received: " + status);
         }
     }
 
     @Override
     public void onCharacteristicRead(BluetoothGatt gatt, BluetoothGattCharacteristic characteristic, int status) {
         if (status == BluetoothGatt.GATT_SUCCESS) {
             broadcastUpdate(BluetoothConstants.ACTION_DATA_AVAILABLE, characteristic);
         }
     }
 
     @Override
     public void onCharacteristicChanged(BluetoothGatt gatt, BluetoothGattCharacteristic characteristic) {
         broadcastUpdate(BluetoothConstants.ACTION_DATA_AVAILABLE, characteristic);
     }
};
 
public void setReConnectAddress(String address) {
     reConnectAddress = address;
}
 
public boolean connect( final String address) {
     if (mBluetoothAdapter == null ) {
     Log.w(TAG, "BluetoothAdapter not initialized." );
     return false ;
     }
     if (address == null ) {
     Log.w(TAG, "Unspecified address." );
     }
     if (mBluetoothDeviceAddress != null && address.equals(mBluetoothDeviceAddress) && mBluetoothGatt != null ) {
     Log.d(TAG, "Trying to use an existing mBluetoothGatt for connection." );
     if (mBluetoothGatt.connect()) {
     mBLEConnectionState = BluetoothConstants.BLE_STATE_CONNECTING;
     return true ;
     } else {
     return false ;
     }
     }
     final BluetoothDevice device = mBluetoothAdapter.getRemoteDevice(address);
     if (device == null ) {
     Log.w(TAG, "Device not found. Unable to connect." );
     return false ;
     }
     mBluetoothGatt = device.connectGatt( this , false , mGattCallback);
     Log.d(TAG, "Trying to create a new connection." );
     mBluetoothDeviceAddress = address;
     Log.d(TAG, "BluetoothDeviceAddress is :" + mBluetoothDeviceAddress);
     mBLEConnectionState = BluetoothConstants.BLE_STATE_CONNECTING;
     return true ;
     }
 
public void close() {
     if (mBluetoothGatt == null ) {
         return ;
     }
     Log.w(TAG, "mBluetoothGatt closed" );
     mBluetoothDeviceAddress = null ;
     disconnect();
     mBluetoothGatt.close();
     mBluetoothGatt = null ;
}

注意看重写的BluetoothGattCallback中onConnectionStateChange()方法中的两个close()方法,此方法调用了mBluetoothGatt.close()用来释放当前的Gatt连接资源,同时在判断为status 133状态时,也就是“status != BluetoothGatt.GATT_SUCCESS”时,在close()方法后会再次调用connect(reConnectAddress)方法,尝试与从设备再次建立连接,直到连接上为止(这里可以做个尝试次数限制)。

其中的reConnectAddress为本次想要连接到的目标从设备蓝牙MAC地址。可以在外部的连接方法中调用setReConnectAddress(String address)方法进行设置。

好了,知道了问题的原因,解决方法其实也就不难了,那么这篇就到这里了。

关于蓝牙方面的内容,希望有相关经验的小伙伴能够在下方留言处多多交流,JACK的机器人在之后的时间也会分享更多相关的经验技巧,感谢大家的支持,谢谢。


本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Android中BLE连接出现“BluetoothGatt status 133”的解决方法 的相关文章

  • App 和设备通过蓝牙连接收发数据

    一 Android 中进行蓝牙开发需要用到的类和执行过程 1 使用BluetoothAdapter startLeScance来扫描设备 2 在扫描到设备的回调函数中的得到BluetoothDevice 对象 并使用Bluetooth st
  • 蓝牙模块AT模式AT指令

    文章目录 进入AT模式的两种方法 HC 05的AT指令 HC 06的AT指令 进入AT模式的两种方法 经过摸索 这里总结两种进入AT模式的方法 与USB转TTL相连后接入电脑 将波特率设置成9600 模块指示灯快闪 这时再按下模块的按钮便进
  • Android蓝牙开发浅析

    由于近期正在开发一个通过蓝牙进行数据传递的模块 在参考了有关资料 并详细阅读了Android的官方文档后 总结了Android中蓝牙模块的使用 更新 之前承诺的蓝牙通讯模块的源码已经放出 详情请点击一下链接 http blog csdn n
  • Android(Java)开发之获取BLE广播包(扫描后获取:广播数据+扫描应答数据+RSSI)

    一 安卓BLE的广播包数据从哪获取 通常 安卓APP读写BLE设备的数据都是建立连接后通过GATT获取或修改 但是 BLE设备向外广播时本身会携带一部分有用信息 如将传感数据存放到广播包的自定义数据段 最近接触的一个iBeacon Eddy
  • 蓝牙之九-AT命令

    AT命令用于HF协议 该命令使参考3GPP 27 007协议 以下是HFP规范 每个命令行只有一个命令 AG侧默认不回显命令 AG使用冗长的格式返回结果 以下字符将被用于AT命令和返回结果格式中
  • Dialog的IDE搭建systermView的方法步骤(DA1469X)

    1 背景 SystemView 是一个可以在线调试嵌入式系统的工具 它可以分析有哪些中断 任务执行了 以及这些中断 任务执行的先后关系 还可以查看一些内核对象持有和释放的时间点 比如信号量 互斥量 事件 消息队列等 这在开发和处理具有多个线
  • 蓝牙AVRCP协议解析

    在这里讨论蓝牙的AVRCP协议 要理解AVRCP 需要先了解AVCTP 我们分两个部分来进行 1 AVCTP协议 2 AVRCP协议 下面先来看AVRCp协议 一 概述 1 定义 AVCTP Audio Video Control Tran
  • 蓝牙之六-A2dp代码调用流程

    上图描述的是蓝牙协议栈 通过该图 查看A2dp的代码在协议栈的调用流程 其分层架构如下 1 蓝牙的系统服务service通过JNI与bluedroid协议栈进行通信 协议栈分为两层 Bluetooth Embedded System BTE
  • 【经典蓝牙】 蓝牙HFP层协议分析

    HFP 概述 HFP概念介绍 HFP Hands Free Profile 是蓝牙免提协议 可以让蓝牙设备对对端蓝牙设备的通话进行控制 例如蓝牙耳机控制手机通话的接听 挂断 拒接 语音拨号等 HFP中蓝牙两端的数据交互是通过定义好的AT指令
  • Android 蓝牙开发(六)hfp连接

    转载请注明出处 http blog csdn net vnanyesheshou article details 71106622 本文已授权微信公众号 fanfan程序媛 独家发布 扫一扫文章底部的二维码或在微信搜索 fanfan程序媛
  • 【低功耗蓝牙】① 蓝牙广播数据格式分析

    摘要 本文章主要讲解了蓝牙的发展史 蓝牙信号 蓝牙广播数据的格式 最后使用ESP32芯片MicroPython固件给出了蓝牙广播的具体代码 是蓝牙初学者很好的参考资料 也可以参考下我在B站的蓝牙视频教程 ESP32教程 第二章 低功耗蓝牙B
  • 蓝牙的知识总结(1)

    1 SoC System on Chip 称为系统级芯片 一个产品 是一个有专用目标的集成电路 其中包含完整系统并有嵌入软件的全部内容 同时它又是一种技术 用以实现从确定系统功能开始 到软 硬件划分 并完成设计的整个过程 从狭义角度讲 它是
  • Bluetooth 蓝牙介绍(七):逻辑链路控制和适配协议规范(L2cap 协议)

    文章目录 前言 L2CAP 特性 前提 术语 常规操作 通道标识符 操作模式 数据包格式 CONNECTION ORIENTED CHANNELS IN BASIC L2CAP MODE CONNECTIONLESS DATA CHANNE
  • STM32+HC-05蓝牙模块学习与使用

    HC 05蓝牙串口通信 HC05模块是一款高性能主从一体蓝牙串口模块 是一种集成蓝牙功能的PCBA板 用于短距离无线通信 十分方便 从某宝商家那里可以看到 蓝牙可以使用多种方法使用 这里我使用的是蓝牙主机连接 所以我们这里需要准备的器件 两
  • 基于snooplog分析蓝牙连接过程,进一步学习蓝牙协议栈

    一 什么是BT snoop log 首先问题 1 为什么远端发来的消息没有收到 2 为什么搜索不到设备 3 为什么连不上 4 总之 研发过程中会遇到很多奇奇怪怪的问题 我们无法通过现象去分析原因 也不可能再过一遍代码吧 这时候需要再某些代码
  • bluez调试笔记

    蓝牙系列 bluez调试笔记 weixin 41069709的博客 CSDN博客 bluezbluez移植https blog csdn net weixin 41069709 article details 125168114 spm 1
  • 蓝牙解析(part7):BLE的连接

    转自Wowo大神的http www wowotech net bluetooth ble connection html 1 前言 了解蓝牙的人都知道 在经典蓝牙中 保持连接 Connection 是一个相当消耗资源 power和带宽 的过
  • 蓝牙之十八- bluetooth pair

    蓝牙之十八 bluetooth pair 在蓝牙核心规范2 1之后 蓝牙配对除了传统的PIN Code Pairing方式外 新增了Secure Simple Pairing配对方式 根据核心规范4 2 简单配对主要有两种目的 蓝牙配对过程
  • pulseaudio使用过程中遇到的问题

    W pulseaudio main c This program is not intended to be run as root unless system is specified E pulseaudio core util c H
  • win32下Qt5BLE蓝牙开发笔记

    BLE简介 BLE蓝牙是蓝牙2 0以上的蓝牙模块 经典蓝牙是蓝牙2 0以下的蓝牙 蓝牙分为客户端和服务器两端 经典蓝牙可以通过socket编程进行客户端与服务器之间的通信 与网络socket相似 BLE蓝牙则无法使用这种方式进行通信 BLE

随机推荐

  • IMU的ROS调试开发工具包:imu_tools

    目录 imu tool包 问题 参数配置便利性问题 实例 调试microstrain 3dm gx5 25 imu 问题 发布的imu姿态与实际imu姿态不一致问题 imu tool包 http wiki ros org imu tools
  • Java串口通信-JSerialComm

    Java串口通信 JSerialComm 目前网上的Java串口通信主要使用RXTXComm 但是这个库已经很久没有更新 最近的更新似乎在2012年 并且与JavaFX集成打包时会出现BUG JSerialComm是一个较新的串口通信库 其
  • 深度学习——权重的初始值

    权重的初始值 权重的初始值十分重要 关系到神经网络的学习是否成功 可以将权重初始值设置为0吗 为了抑制过拟合 提高泛化能力 采用权值衰减的方法 它是一种以减小权重参数的值为目的进行学习的方法 在误差反向传播法中 所有的权重值都会进行相同的更
  • truss 命令

    truss 命令用途跟踪进程的系统调用 动态装入的用户级函数调用 接收的信号和造成的机器故障 语法truss f c a l d D e i t x Syscall s Signal m Fault r w FileDescriptor u
  • gitlab 搭建与使用

    gitlab 搭建与使用 参考博客链接 https blog csdn net weixin 38912024 article details 82383312 https blog csdn net hao5335156 article
  • JasperReport6.x 提示:No markuo processor factory specified for ““ markup

    Jaspersoft Studio 提示如下错误信息 No markuo processor factory specified for markup 解决方法 jrxml 切换至源码模式 查询makeup 属性字段标签 将内容编辑为 no
  • ssh: connect to host port 22: No route to host问题

    笔记本没电自动关机 但是虚拟机是开启状态 开机之后发现连接不上其中一个虚拟机 出现了ssh connect to host port 22 No route to host问题提示 后来检查防火墙和网都没有问题 使用ifconfig命令后发
  • ie 使用window.open页面报错

    window open url 打开新页面是如果要通过地址栏来传参要注意 var qt qt teachMaterialDealInfo bookids selectBooks qt teachMaterialDealInfo writer
  • vba文字型的值转换成数值类型

    今天在做表的时候 发现数组中提取出来的是文字类型 需要转换成数值类型 网上搜索到了字符转换的方法 超级实用 记录下来 如果用mid函数提取的字符范围增加 可参照VBA中的帮助 类型转换函数 CBool expression CByte ex
  • 正则表达式匹配数字、字母和汉字等各类汇总

    最近在开发中遇到一个需求是只匹配字母和汉字 于是在网上找了一个比较全的记录一下 日后再用 正则表达式来匹配规范一段文本中的特定种类字符 下面是对常用的正则匹配做了一个归纳整理 1 匹配中文 u4e00 u9fa5 2 英文字母 a zA Z
  • unity code-动作系统Animator

    AnimatorOverrideController是运行时版本 可以运行时更改controller AnimatorController是编辑器版本 不能运行时修改 所以运行时只能通过用AnimatorOverrideController
  • 自动化测试框架之RobotFramework进行接口测试

    首先在进行接口测试前分析一下如何进行接口请求的发送或进行接口请求发送包含哪些内容 那么就需要对http请求包含的内容 请求行 协议版本 URL 请求方法 请求头 host content Type Accept等 请求体 发送请求需要带的参
  • 关于SSM框架体会和小结

    前言 使用SSM spring SpringMVC和Mybatis 已经有三个多月了 项目在技术上已经没有什么难点了 基于现有的技术就可以实现想要的功能 当然肯定有很多可以改进的地方 之前没有记录SSM整合的过程 只写了个搭建Maven过程
  • 2021-02-08

    学习目标 综合练习题 10道经典题目 天池龙珠计划SQL训练营 学习内容 10道sql练习题 学习产出 1 说实话自己写不全 还是翻看了答案才知道 2 自己装了sql建库建表 导入数据特别慢 电脑配置不够
  • 【Vue学习笔记】- 事件的基本使用

    总结 1 使用 v on xxx 或 xxx 绑定事件 其中xxx是事件名 2 事件的回调需要配置在methods对象中 最终会在vm上 3 methods中配置的函数 不要用箭头函数 否则this就不是vm了 4 methods中配置的函
  • 【软件测试】自动化测试战零基础教程——Python自动化从入门到实战(二)

    软件测试知识持续更新中 第二章 python webdriver 环境搭建 第一节 环境搭建 准备工具如下 windows 环境安装 linux 环境安装 第二节 使用 IDLE 来编写 python 有效地使用 IDLE 第三节 第一个自
  • map 遍历时 删除元素

    import java io File import java io IOException import java util HashMap import java util Iterator import java util Map i
  • JMeter与Selenium WebDriver集成的价值

    JMeter作为性能或接口测试工具 用在UI测试上多少属于不伦不类 而Selenium WebDriver作为如今最主流的UI开源测试工具 得到许多人的认可 那么为什么有人要把JMeter与Selenium WebDriver进行结合呢 我
  • Android SwipeRefreshLayout 嵌套NestedScrollView 偶尔无法在下拉刷新的问题

    第一次接触谷歌的下拉刷新不太会用导致的问题 问题是在下拉刷新的时候长按不松开继续往下滑导致下拉刷新呗关闭 简单来说就是在下拉刷新的时候继续往上滑动触发ScrollView的滑动事件后关闭下拉刷新 如果是下拉刷新在运行中被关闭setEnabl
  • Android中BLE连接出现“BluetoothGatt status 133”的解决方法

    http www loverobots cn android ble connection solution bluetoothgatt status 133 html 前 言 最近的工作方向一直在低功耗蓝牙方面 也就是BLE Blueto