常见的蓝牙标准有2.0和4.0。
| |特点 |
|------------|
| 2.0 | 1.适用于数据量比较大得传输,比如音乐、语音
2.IOS开发中,要求设备是经过MFI认证|
| 4.0 | 1.适用于实时性比较高的数据传输,比如遥控类的鼠标、键盘,传感设备的心跳计、血压计
2.功耗低,距离短,轻量级|
注意:
一般我们说的蓝牙4.0都支持2.0和4.0,只是我们开发过程中只负责4.0标准的部分。
一、基本知识
1.CoreBluetooth框架是ble4.0开发使用的框架
2.CoreBluetooth主要内容有两个:peripheral(外设模式)和central(中心模式)。现在我写的是central内容。
二、外设模式关键操作
1.创建中心角色(centralManger)
2.扫描外设(peripheral)
注意:
该步骤需确认centralManager.state==CBCentralManagerStatePoweredOn
3.持有并连接外设(connect)
4.扫描外设的服务(service)
注意:该步骤需确认didConnectPeripheral
方法调用成功
5.扫描服务的特征(characteristic)
注意:该步骤需要确认didDiscoverServices
方法回调成功
6.操作特征
注意:该操作需确认didDiscoverCharacteristicsForService
成功回调
6.1.重新读取特征值(read)
6.2.往特征里写入值(write)
6.3.订阅特征(notifying)
6.4.扫描描述(descriptor)
7.断开连接,停止扫描
三、代码
#import "BLETool.h"
#import <CoreBluetooth/CoreBluetooth.h>
@interface BLETool()<CBCentralManagerDelegate,CBPeripheralDelegate>
@property (strong, nonatomic) CBCentralManager *centralManager;
@property (strong, nonatomic) CBPeripheral *peripheral;
@end
@implementation BLETool
- (instancetype)init{
if (self = [super init]) {
//1.创建中心管理角色。
/**
queue为nil表示默认主线程
*/
self.centralManager = [[CBCentralManager alloc] initWithDelegate:self queue:nil];
}
return self;
}
/**
* -- 必须实现的代理,用来返回创建的centralManager的状态。
* -- 注意:必须确认当前是CBCentralManagerStatePoweredOn状态才可以调用扫描外设的方法:
scanForPeripheralsWithServices
*/
- (void)centralManagerDidUpdateState:(CBCentralManager *)central{
switch (central.state) {
case CBCentralManagerStateUnknown:
NSLog(@">>>CBCentralManagerStateUnknown");
break;
case CBCentralManagerStateResetting:
NSLog(@">>>CBCentralManagerStateResetting");
break;
case CBCentralManagerStateUnsupported:
NSLog(@">>>CBCentralManagerStateUnsupported");
break;
case CBCentralManagerStateUnauthorized:
NSLog(@">>>CBCentralManagerStateUnauthorized");
break;
case CBCentralManagerStatePoweredOff:
NSLog(@">>>CBCentralManagerStatePoweredOff");
break;
case CBCentralManagerStatePoweredOn:
{
NSLog(@">>>CBCentralManagerStatePoweredOn");
//2.开始扫描周围的外设。
/*
-- 两个参数为Nil表示默认扫描所有可见蓝牙设备。
-- 注意:第一个参数我一开始以为是扫描指定外设的,那么使用外设的identifier就可以了。后来发现不是,这个参数是用来扫描有指定服务的外设。然后有些外设的服务是相同的,比如都有FFF5服务,那么都会发现;而有些外设的服务是不可见的,就会扫描不到设备。
-- 成功扫描到外设后调用didDiscoverPeripheral
*/
[self.centralManager scanForPeripheralsWithServices:nil options:nil];
}
break;
default:
break;
}
}
#pragma mark 发现外设
- (void)centralManager:(CBCentralManager *)central didDiscoverPeripheral:(CBPeripheral *)peripheral advertisementData:(NSDictionary<NSString *,id> *)advertisementData RSSI:(NSNumber *)RSSI{
if ([peripheral.name isEqualToString:@"TimesBasy-BLE"]||[peripheral.name isEqualToString:@"TimesBaby-BLE"]) {
//注意:想要对指定peripheral进行后续操作,一定要保存这个外设对象。否则centralManager会认为你对指定peripheral不感兴趣,这样你即不能再扫描到这个指定peripheral,也不能对他进行后续操作(比如回调didConnectPeripheral)
self.peripheral = peripheral;
//3.连接外设
/**
-- 一个中心管理角色可以连接多个外设,但是一个外设只能被一个角色连接,外设被连接后就不能能再被扫描到
-- 连接成功回调didConnectPeripheral,失败回调didFailToConnectPeripheral,取消连接回调didDisconnectPeripheral
*/
[self.centralManager connectPeripheral:peripheral options:nil];
}
}
#pragma mark 连接外设--成功
- (void)centralManager:(CBCentralManager *)central didConnectPeripheral:(CBPeripheral *)peripheral{
//连接成功后停止扫描,节省内存
[central stopScan];
peripheral.delegate = self;
//4.扫描外设的服务
/**
-- 外设的服务、特征、描述等方法是CBPeripheralDelegate的内容,所以要先设置代理peripheral.delegate = self
-- 参数表示你关心的服务的UUID,比如我关心的是"FFF0",参数就可以为@[[CBUUID UUIDWithString:@"FFF0"]].那么didDiscoverServices方法回调内容就只有这两个UUID的服务,不会有其他多余的内容,提高效率。nil表示扫描所有服务
-- 成功发现服务,回调didDiscoverServices
*/
[peripheral discoverServices:nil];
}
#pragma mark 连接外设——失败
- (void)centralManager:(CBCentralManager *)central didFailToConnectPeripheral:(CBPeripheral *)peripheral error:(NSError *)error{
}
#pragma mark 取消与外设的连接回调
- (void)centralManager:(CBCentralManager *)central didDisconnectPeripheral:(CBPeripheral *)peripheral error:(NSError *)error{
}
#pragma mark 发现服务回调
- (void)peripheral:(CBPeripheral *)peripheral didDiscoverServices:(NSError *)error{
for (CBService *service in peripheral.services) {
//5.扫描指定服务的特征
/**
-- 第一个参数是数组,表示扫描指定服务下那些我关心的特征的UUID。
-- 成功扫描到特征后回调didDiscoverCharacteristicsForService
*/
[peripheral discoverCharacteristics:nil forService:service];
}
}
#pragma mark 发现特征回调
/**
-- 发现特征后,可以根据特征的properties进行:读readValueForCharacteristic、写writeValue、订阅通知setNotifyValue、扫描特征的描述discoverDescriptorsForCharacteristic。
-- 注意:实际开发中根据文档来确定对指定characteristic的操作。因为可能你得蓝牙设备的特征的properties不在下面之列。比如properties = 0x18
说明:我注视的那些不叫重要,其他的我也不知道
CBCharacteristicPropertyBroadcast = 0x01,
//允许读
CBCharacteristicPropertyRead = 0x02,
//允许写,但不会回应(不会调用didWriteValueForCharacteristic)
CBCharacteristicPropertyWriteWithoutResponse = 0x04,
//允许写,但会回应
CBCharacteristicPropertyWrite = 0x08,
//允许通知
CBCharacteristicPropertyNotify = 0x10,
//允许通知。和上面分属两种不同的通知类型
CBCharacteristicPropertyIndicate = 0x20,
CBCharacteristicPropertyAuthenticatedSignedWrites = 0x40,
CBCharacteristicPropertyExtendedProperties = 0x80,
CBCharacteristicPropertyNotifyEncryptionRequired NS_ENUM_AVAILABLE(NA, 6_0) = 0x100,
CBCharacteristicPropertyIndicateEncryptionRequired
*/
- (void)peripheral:(CBPeripheral *)peripheral didDiscoverCharacteristicsForService:(CBService *)service error:(NSError *)error{
//一般开发时我们都应该有蓝牙文档的,需要读、写、订阅的特征都说说ing,所以直接根据对应的特征的UUID操作就可以,不用根据properties。
for (CBCharacteristic *characteristic in service.characteristics) {
if ([characteristic.UUID isEqual:[CBUUID UUIDWithString:@"FFF0"]]) {
//读。重新获取characteristic的值
/**
-- 阅读文档查看需要对该特征的操作
-- 读取成功回调didUpdateValueForCharacteristic
*/
[peripheral readValueForCharacteristic:characteristic];
}
if ([characteristic.UUID isEqual:[CBUUID UUIDWithString:@"FFF1"]]) {
//写。
/**
-- 阅读文档查看需要对该特征的操作
-- type:CBCharacteristicWriteWithResponse表示写入成功会回调didWriteValueForCharacteristic
*/
NSData *data = [NSData new];
[peripheral writeValue:data forCharacteristic:characteristic type:CBCharacteristicWriteWithResponse];
}
if ([characteristic.UUID isEqual:[CBUUID UUIDWithString:@"FFF5"]]) {
//订阅通知
/**
-- 通知一般在蓝牙设备状态变化时会发出,比如蓝牙音箱按了上一首或者调整了音量,如果这时候对应的特征被订阅,那么app就可能收到通知
-- 阅读文档查看需要对该特征的操作
-- 订阅成功后回调didUpdateNotificationStateForCharacteristic
-- 订阅后characteristic的值发生变化会发送通知到didUpdateValueForCharacteristic
-- 取消订阅:设置setNotifyValue为NO
*/
[peripheral setNotifyValue:YES forCharacteristic:characteristic];
}
//扫描描述
/**
-- 进一步提供characteristic的值的相关信息。(因为我项目里没有的特征没有进一步描述,所以我也不怎么理解)
-- 当发现characteristic有descriptor,回调didDiscoverDescriptorsForCharacteristic
*/
[peripheral discoverDescriptorsForCharacteristic:characteristic];
}
}
#pragma mark 数据写入特征回调
- (void)peripheral:(CBPeripheral *)peripheral didWriteValueForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error{
}
#pragma mark 获取特征的值
/**
-- peripheral调用readValueForCharacteristic成功后会回调该方法
-- peripheral调用setNotifyValue后,特征发出通知也会调用该方法
*/
- (void)peripheral:(CBPeripheral *)peripheral didUpdateValueForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error{
}
#pragma mark 订阅通知回调
- (void)peripheral:(CBPeripheral *)peripheral didUpdateNotificationStateForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error{
}
#pragma mark 发现descriptors回调
- (void)peripheral:(CBPeripheral *)peripheral didDiscoverDescriptorsForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error{
}
#pragma mark 断开连接
- (void)disConnectPeripheral{
/**
-- 断开连接后回调didDisconnectPeripheral
-- 注意断开后如果要重新扫描这个外设,需要重新调用[self.centralManager scanForPeripheralsWithServices:nil options:nil];
*/
[self.centralManager cancelPeripheralConnection:self.peripheral];
}
#pragma mark 停止扫描外设
- (void)stopScanPeripheral{
[self.centralManager stopScan];
}
@end