常见的蓝牙标准有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@interface BLETool() @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 *)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