1、通過掃描尋找服務(wù)
//允許iOS設(shè)備作為客戶端從一個藍牙設(shè)備讀取數(shù)據(jù)
self.manager = [[CBCentralManager alloc] initWithDelegate:self queue:nil options:@{CBCentralManagerOptionShowPowerAlertKey:@(NO)}];
首先掃描外圍設(shè)備
//從一個藍牙設(shè)備讀取數(shù)據(jù) 開始掃描藍牙 是否允許中央設(shè)備多次收到曾經(jīng)監(jiān)聽到的設(shè)備的消息,這樣來監(jiān)聽外圍設(shè)備聯(lián)接的信號強度,以決定是否增大廣播強度,為YES時會多耗電
[self.manager scanForPeripheralsWithServices:@[[CBUUID UUIDWithString:_serviceUUID]] options:@{CBCentralManagerScanOptionAllowDuplicatesKey : [NSNumber numberWithBool:YES]}];
scanForPeripheralsWithServices:方法使用了一個服務(wù)數(shù)組作為參數(shù),掃描周邊地區(qū)廣播這些服務(wù)的外圍設(shè)備,如果使用Nil作為參數(shù),會掃描所有可用的外圍設(shè)備,不過這樣會很慢。
CBCentralManager這個類有一個委托CBCentralManagerDelegate這個委托可以通知你已發(fā)現(xiàn)的外圍設(shè)備、服務(wù)、服務(wù)的特性和數(shù)值變化。如果你的藍牙么有打開,會在CBCentralManagerDelegate的centralManagerDidUpdateState方法中收到回調(diào),這個方法也要開始掃描外圍設(shè)備。
#pragma mark - CBCentralManagerDelegate
/**
*當藍牙狀態(tài)改變的時候就會調(diào)用這個方法
*/
- (void)centralManagerDidUpdateState:(CBCentralManager *)central{
switch (central.state) {
case CBManagerStatePoweredOn:
//正在重置狀態(tài) 可以使用GCD
case CBManagerStateResetting:
{
dispatch_async(dispatch_get_main_queue(), ^{
if (self.connectState == DoorOpenStateScanning)
[self starScanPeripheral];
});
}
break;
//
case CBManagerStateUnknown:
{
}
break;
default:
{
[self stopConnect];
}
break;
}
}
找到外圍設(shè)備后,通過CBCentralManagerDelegate的didDiscoverPeripheral:方法可以獲取外圍設(shè)備的詳細信息。
2.連接設(shè)備
找到外圍設(shè)備后,下一步就是連接外圍設(shè)備并發(fā)現(xiàn)他提供的服務(wù)。
/**
*發(fā)現(xiàn)服務(wù)
*/
- (void)centralManager:(CBCentralManager *)central didDiscoverPeripheral:(CBPeripheral *)peripheral advertisementData:(NSDictionary *)advertisementData RSSI:(NSNumber *)RSSI
{
//處理業(yè)務(wù)邏輯
//連接外圍設(shè)備
[central connectPeripheral:peripheral options:nil];
}
在連接外圍設(shè)備之前應(yīng)該先保留之,因為ARC編譯器會釋放外圍設(shè)備對象而導(dǎo)致其無法連接。可以加入到一個數(shù)組里面來保留。
3.直接獲取外圍設(shè)備
如果你知道外圍設(shè)備的標識符,可以使用這個方法掃描。這個方法是iOS7加入的。
先把外圍設(shè)備列表加到數(shù)組,保存這個數(shù)組,每次掃描設(shè)備之前先嘗試連接已知外圍設(shè)備是個好習(xí)慣。掃描比較費電,應(yīng)該避免。拿到外圍設(shè)備的指針后,就可以直接連接了。
- (NSArray<CBPeripheral *> *)retrievePeripheralsWithIdentifiers:(NSArray<NSUUID *> *)identifiers NS_AVAILABLE(NA, 7_0);
4.發(fā)現(xiàn)服務(wù)
建立連接的嘗試可能成功也可能失敗,如果成功就會調(diào)用這個委托方法。下一步是發(fā)現(xiàn)外圍設(shè)備提供的服務(wù)。
/**
*連接成功之后調(diào)用的方法,然后開始掃描
*peripheral 成功 掃描外設(shè)中的服務(wù)和特征,連接上外圍設(shè)備的時候回調(diào)用這個方法
* NSLog(@">>>連接到名稱為(%@)的設(shè)備-成功",peripheral.name);
*/
- (void)centralManager:(CBCentralManager *)central didConnectPeripheral:(CBPeripheral *)peripheral
{
self.peripheral = peripheral;
_isConnect = YES;
self.connectState = DoorOpenStateConnected;
//設(shè)置的peripheral委托CBPeripheralDelegate
[peripheral setDelegate:self]; //查找服務(wù)
//掃描外設(shè)Services,成功后會進入方法
[peripheral discoverServices:nil];
}
外圍設(shè)備提供的服務(wù)列表是通過CBPeripheralDelegate一個委托方法通知的,從外圍設(shè)備的services屬性中可以獲取服務(wù)列表。
#pragma mark - CBPeripheralDelegate
/**
* 掃描外設(shè)Services,成功后會進入方法
* 發(fā)現(xiàn)外圍設(shè)備的服務(wù)會來到該方法(掃描到服務(wù)之后直接添加peripheral的services)
*/
- (void)peripheral:(CBPeripheral *)peripheral didDiscoverServices:(NSError *)error
{
if (error)
{
if ([self.delegate respondsToSelector:@selector(openDoorFailed:error:)]) {
[self.delegate openDoorFailed:self.curDoor error:error];
}
[self stopConnect];
return;
}
BOOL bFind = NO;
for (CBService *service in peripheral.services)
{
//發(fā)現(xiàn)服務(wù)
if ([service.UUID.UUIDString isEqualToString:@"FFF0"])
{
NSLog(@"====%@------%@+++++++",service.UUID.UUIDString,self.peripheral.identifier);
// characteristicUUIDs : 可以指定想要掃描的特征(傳nil,掃描所有的特征)
_service = service;
[peripheral discoverCharacteristics:nil forService:service];
bFind = YES;
break;
}
}
if (!bFind) {
[self stopConnect];
if ([self.delegate respondsToSelector:@selector(openDoorFailed:error:)]) {
NSError *error1 = [NSError errorWithDomain:@"openDoorFailed" code:LeEGNotDiscoverServices userInfo:@{NSLocalizedDescriptionKey:@"沒有找到指定藍牙服務(wù)"}];
[self.delegate openDoorFailed:self.curDoor error:error1];
}
}
}
5.發(fā)現(xiàn)特征
特性:打開/關(guān)閉服務(wù)或者是讀取服務(wù)的當前值。(如上)返回已經(jīng)發(fā)現(xiàn)的特性的委托方法。
/**
* 找到了設(shè)備的服務(wù)然后掃描特征
*/
- (void)peripheral:(CBPeripheral *)peripheral didDiscoverCharacteristicsForService:(CBService *)service error:(nullable NSError *)error {
NSLog(@"發(fā)現(xiàn)外圍設(shè)備的特征");
for (CBCharacteristic *characteristic in service.characteristics)
{
NSLog(@"====%@------+",characteristic.UUID.UUIDString);
//發(fā)現(xiàn)特征
if ([characteristic.UUID isEqual:[CBUUID UUIDWithString:@"FFF1"]]) {
// NSLog(@"監(jiān)聽:%@",characteristic);//監(jiān)聽特征
// 拿到特征,和外圍設(shè)備進行交互
[self.peripheral setNotifyValue:YES forCharacteristic:characteristic];
self.charWrite = characteristic;
} else if ([characteristic.UUID isEqual:[CBUUID UUIDWithString:@"FFF4"]]) {
[self.peripheral setNotifyValue:YES forCharacteristic:characteristic];
self.charRead = characteristic;
}
}
if (self.charRead && self.charWrite) {
if (![self need3DES:self.curDoor]) {
[self sendOpenDoorCommand];
} else {
// 拿到特征,和外圍設(shè)備進行交互 保存寫的特征
[self WriteValue:@"BB"];
self.connectState = DoorOpenStateWaitKey;
}
} else {
if ([self.delegate respondsToSelector:@selector(openDoorFailed:error:)]) {
NSError *error1 = [NSError errorWithDomain:@"openDoorFailed" code:LeEGNotDiscoverCharacteristics userInfo:@{NSLocalizedDescriptionKey:@"特征值缺失"}];
[self.delegate openDoorFailed:self.curDoor error:error1];
}
[self stopConnect];
return;
}
}
在下面的方法里面,我們會接收到數(shù)據(jù)更新或者改變的消息
/**
* 設(shè)置通知,數(shù)據(jù)通知會進入:didUpdateValueForCharacteristic方法
*/
- (void)peripheral:(CBPeripheral *)peripheral didUpdateValueForCharacteristic:(CBCharacteristic *)characteristic error:(nullable NSError *)error {}
數(shù)值變化已NSData形式發(fā)送過來的。需要根據(jù)情況進行轉(zhuǎn)換實際的數(shù)據(jù)。
6.創(chuàng)建自己的外圍設(shè)備
CoreBluetooth.framework在iOS7中增加了 CBPeripheralManager 和 CBCentral兩個類。