一 :前言
对于刚接触蓝牙的萌新来说,直接用CoreBluetooch会感觉语句很复杂凌乱写的到处都是,不优雅。经常是10几个委托方法,而且还得经常在委托中调用方法再进其他委托,整体结构层次就很不清晰。所以对于我这样的蓝牙萌新来说,选择BabyBluetooth来进行蓝牙开发还是很有帮助的。
二:BabyBluetooth介绍
BabyBluetooth是一个最简单易用的蓝牙库,基于CoreBluetooth的封装,并兼容ios和mac osx。
BabyBluetooth的优点:
1:基于原生CoreBluetooth框架封装的轻量级的开源库,可以帮你更简单地使用CoreBluetooth API。
2:CoreBluetooth所有方法都是通过委托完成,代码冗余且顺序凌乱。BabyBluetooth使用block方法,可以重新按照功能和顺序组织代码,并提供许多方法减少蓝牙开发过程中的代码量。
3:链式方法体,代码更简洁、优雅。
4:通过channel切换区分委托调用,并方便切换
5:便利的工具方法
6:完善的文档,且项目处于活跃状态,不断的更新中
7:github上star最多的纯Bluetooch类库(非PhoneGap和SensorTag项目)
三: BabyBluetooth集成
在我自己的项目中,我选择的是用cocoapods集成,不得不说,用cocoapods集成比自己集成要方便很多,集成步骤如下:
1:把pod 'BabyBluetooth', '~> 0.6.0'添加到你的Podfile中。
2.在命令行中执行pod install。
3.将CocoaPods生成的.xcworkspace运用到你的开发项目中即可。
四:BabyBluetooth使用详解
蓝牙开发一般有如下五个步骤
1、扫描外设
2、发现外设
3、连接外设
4、获取外设读写的特征
5、获取外设发出的信息和向外设发信息
接下来会以这五个步骤的顺序来介绍BabyBluetooth的使用;
1、扫描外设
初始化好BabyBluetooth后就可以用起提供的链式语法self.baby.scanForPeripherals().begin();来开启扫描了,同样也可以用其链式语法来停止扫描:self.baby.scanForPeripherals().stop(30);也可以吧开始和停止写在一起比如:baby.scanForPeripherals().begin().stop(4);即开始扫描后4秒停止扫描。当然BabyBluetooth也提供了判断蓝牙是否开启成功的block回调,值得注意的是这里并不需要在蓝牙开启的回调里判断CBCentralManagerStatePoweredOn状态再开启扫描,而是在设置代理后可以直接开启扫描,回调block如下:
2、发现外设
扫描之后如果发现了外设会进入发现外设的代理,如下
参数说明:1. peripheral为扫描发现的外设。 2. advertisementData为外设广播包携带数据,主要获取有几个服务显示出来,只要用这个key:kCBAdvDataServiceUUIDs。3. RSSI为外设信号。具体解析可以参照如下
3. 选择你的目标外设并开始连接外设:
baby.having(currPeripheral).enjoy();
不过为了图方便,我一般会直接连接外设加发现服务发现特征等等功能直接来一套,如下:
连接成功后的回调
在babydemo里代理都加上了Channel,这里我个人看法是如非必要不要加,加了之后后续要连接多个蓝牙就很难处理的。在我自己的项目里我就没有加Channel。
4、获取外设读写的特征
在连接成功后,可以在发现发现特征值的block回调里选择你需要用到的读特征和写特征(Characteristics),回调如下:
[
然后根据服务的uuid,以及特征的uuid来找到你需要的特征值,具体蓝牙的服务ID和读写通道可以参考你们目标蓝牙的蓝牙协议
[self.baby setBlockOnDiscoverCharacteristics:^(CBPeripheral *peripheral, CBService *service, NSError *error) {
NSLog(@"===service name:%@",service.UUID);
for (CBCharacteristic *c in service.characteristics)
{
if ([c.UUID.UUIDString isEqualToString:@"FFE1"]) {
}
if ([c.UUID.UUIDString isEqualToString:@"F000C0E1-0451-4000-B000-000000000000"])
{
weakSelf.readcharacteristic = c;
[weakSelf addNotifyWIth:c];
}
if ([c.UUID.UUIDString isEqualToString:@"F000C0E2-0451-4000-B000-000000000000"])
{
weakSelf.writecharacteristic = c;
[weakSelf addNotifyWIth:c];
}
}
}
}]
一般蓝牙的读写通道(Characteristics)都是分开的,少数蓝牙会读写用同一个通道,在发现目标通道后,我们要利用写通道来向蓝牙写入数据,用读通道来读取蓝牙发来的包。
5、获取外设发出的信息和向外设发信息
1)写数据
写数据比较方便,直接调用以下方法就可以:
但是这里的data是有要求的,不能直接用字符串utf8转成data,一般写入蓝牙的都是16进制字符串,在此提供一个16进制字符串转data的方法,方法如下
除此之外,写数据还要注意的是type,type有两种,一种是有回应,一种是无回应,无回应的写入操作type写成了有回应会导致写入失败,报错写入不允许
2 )读数据
凡是从蓝牙传过来的数据都要经过这个回调,简单的说这个方法就是你拿数据的唯一方法),这是一个系统方法,BabyBluetooth对这个方法进行了如下封装
封装出来两个block,一个是
另外一个是
根据上面BabyBluetooth代码来看notify的优先级是高于写数据成功的block,而且如果这个特征你监听了它,由于有个return,后面那个block就不会走了。
在我的项目中,我是通过notify来获取蓝牙回包的,在notify的回调block里你可以通过如下方式获取蓝牙回包内容,并进行后续操作了,
到此,BabyBluetooth的完整开发流程就已经基本结束了。
五:使用BabyBluetooth的注意事项
1、在页面销毁的时候要断开蓝牙连接释放蓝牙资源。
2、点击蓝牙连接的时候,这是如果立马跳转下个页面去收发数据,会崩溃,这是因为在连接蓝牙的时候读取特征没完成就跳转了。解决方法是等特征读完再跳转,可以加一个loading
3、第一次连接断开之后再连接的时候接收数据的代理直接触发,这个时候数据处理如果不当的话会crash,解决方法是:在离开收发数据那个页面的时候加一句代码重置那个block
_baby setBlockOnReadValueForCharacteristic:^(CBPeripheral*peripheral, CBCharacteristic *characteristic, NSError *error) {}];