iOS藍牙開發(fā)

CoreBluetooth詳解

CoreBluetooth框架的核心其實是兩個東西,peripheral和central, 可以理解成外設(shè)和中心。

圖中兩組api分別對應不同的業(yè)務(wù)場景,左側(cè)叫做中心模式,就是以你的app作為中心,連接其他的外設(shè)的場景,而右側(cè)稱為外設(shè)模式,使用手機作為外設(shè)別其他中心設(shè)備操作的場景

image

關(guān)于藍牙開發(fā)的一些重要的理論概念:

  1、服務(wù)(services):藍牙外設(shè)對外廣播的時候一定會有一個服務(wù),有些時候也可以是有多個服務(wù),服務(wù)下面包含一些特性,服務(wù)可以理解成一個模塊的窗口;
  2、特征(characteristic):特征存在于服務(wù)下面的,一個服務(wù)下面可以有多個特征,特征可以理解成具體實現(xiàn)功能的窗口,一般的特性都會有value,也就是特征值,是特征和外界交互的最小單位;
  3、UUID:藍牙上的唯一標示符,為了區(qū)分不同設(shè)備、服務(wù)及特征,就用UUID來表示。

CBCentralMannager 中心模式

以手機(app)作為中心,連接其他外設(shè)的場景。詳細流程如下:

步驟1.建立一個Central Manager實例進行藍牙管理

步驟2.搜索外圍設(shè)備

步驟3.連接外圍設(shè)備

步驟4.獲得外圍設(shè)備的服務(wù)

步驟5.獲得服務(wù)的特征

步奏6.從外圍設(shè)備讀數(shù)據(jù)(直接讀取和訂閱兩種方法)

步驟7.給外圍設(shè)備發(fā)送數(shù)據(jù)

#import "ViewController.h"
#import 
#define kPeripheralName @"Kenshin Cui's Device" //外圍設(shè)備名稱
#define kServiceUUID @"C4FB2349-72FE-4CA2-94D6-1F3CB16331EE" //服務(wù)的UUID
#define kCharacteristicUUID @"6A3E4B28-522D-4B3B-82A9-D5E2004534FC" //特征的UUID

@interface ViewController ()

@property (strong,nonatomic) CBPeripheralManager *peripheralManager;//外圍設(shè)備管理器

@property (strong,nonatomic) NSMutableArray *centralM;//訂閱此外圍設(shè)備特征的中心設(shè)備

@property (strong,nonatomic) CBMutableCharacteristic *characteristicM;//特征
@property (weak, nonatomic) IBOutlet UITextView *log; //日志記錄

@end
@implementation ViewController
#pragma mark - 控制器視圖事件
- (void)viewDidLoad {
    [super viewDidLoad];
}

#pragma mark - UI事件
- (IBAction)startClick:(UIBarButtonItem *)sender {
    //創(chuàng)建中心設(shè)備管理器并設(shè)置當前控制器視圖為代理
    _centralManager=[[CBCentralManager alloc]initWithDelegate:self queue:nil];
}

#pragma mark - CBCentralManager代理方法
//中心服務(wù)器狀態(tài)更新后
-(void)centralManagerDidUpdateState:(CBCentralManager *)central{
    switch (central.state) {
        case CBPeripheralManagerStatePoweredOn:
            NSLog(@"BLE已打開.");
            [self writeToLog:@"BLE已打開."];
            //掃描外圍設(shè)備
//            [central scanForPeripheralsWithServices:@[[CBUUID UUIDWithString:kServiceUUID]] options:@{CBCentralManagerScanOptionAllowDuplicatesKey:@YES}];
            [central scanForPeripheralsWithServices:nil options:@{CBCentralManagerScanOptionAllowDuplicatesKey:@YES}];
            break;

        default:
            NSLog(@"此設(shè)備不支持BLE或未打開藍牙功能,無法作為外圍設(shè)備.");
            [self writeToLog:@"此設(shè)備不支持BLE或未打開藍牙功能,無法作為外圍設(shè)備."];
            break;
    }
}
/**
 *  發(fā)現(xiàn)外圍設(shè)備
 *
 *  @param central           中心設(shè)備
 *  @param peripheral        外圍設(shè)備
 *  @param advertisementData 特征數(shù)據(jù)
 *  @param RSSI              信號質(zhì)量(信號強度)
 */
-(void)centralManager:(CBCentralManager *)central didDiscoverPeripheral:(CBPeripheral *)peripheral advertisementData:(NSDictionary *)advertisementData RSSI:(NSNumber *)RSSI{
    NSLog(@"發(fā)現(xiàn)外圍設(shè)備...");
    [self writeToLog:@"發(fā)現(xiàn)外圍設(shè)備..."];
    //停止掃描
    [self.centralManager stopScan];
    //連接外圍設(shè)備
    if (peripheral) {
        //添加保存外圍設(shè)備,注意如果這里不保存外圍設(shè)備(或者說peripheral沒有一個強引用,無法到達連接成功(或失?。┑拇矸椒?,因為在此方法調(diào)用完就會被銷毀
        if(![self.peripherals containsObject:peripheral]){
            [self.peripherals addObject:peripheral];
        }
        NSLog(@"開始連接外圍設(shè)備...");
        [self writeToLog:@"開始連接外圍設(shè)備..."];
        [self.centralManager connectPeripheral:peripheral options:nil];
    }

}
//連接到外圍設(shè)備
-(void)centralManager:(CBCentralManager *)central didConnectPeripheral:(CBPeripheral *)peripheral{
    NSLog(@"連接外圍設(shè)備成功!");
    [self writeToLog:@"連接外圍設(shè)備成功!"];
    //設(shè)置外圍設(shè)備的代理為當前視圖控制器
    peripheral.delegate=self;
    //外圍設(shè)備開始尋找服務(wù)
    [peripheral discoverServices:@[[CBUUID UUIDWithString:kServiceUUID]]];
}
//連接外圍設(shè)備失敗
-(void)centralManager:(CBCentralManager *)central didFailToConnectPeripheral:(CBPeripheral *)peripheral error:(NSError *)error{
    NSLog(@"連接外圍設(shè)備失敗!");
    [self writeToLog:@"連接外圍設(shè)備失敗!"];
}

#pragma mark - CBPeripheral 代理方法
//外圍設(shè)備尋找到服務(wù)后
-(void)peripheral:(CBPeripheral *)peripheral didDiscoverServices:(NSError *)error{
    NSLog(@"已發(fā)現(xiàn)可用服務(wù)...");
    [self writeToLog:@"已發(fā)現(xiàn)可用服務(wù)..."];
    if(error){
        NSLog(@"外圍設(shè)備尋找服務(wù)過程中發(fā)生錯誤,錯誤信息:%@",error.localizedDescription);
        [self writeToLog:[NSString stringWithFormat:@"外圍設(shè)備尋找服務(wù)過程中發(fā)生錯誤,錯誤信息:%@",error.localizedDescription]];
    }
    //遍歷查找到的服務(wù)
    CBUUID *serviceUUID=[CBUUID UUIDWithString:kServiceUUID];
    CBUUID *characteristicUUID=[CBUUID UUIDWithString:kCharacteristicUUID];
    for (CBService *service in peripheral.services) {
        if([service.UUID isEqual:serviceUUID]){
            //外圍設(shè)備查找指定服務(wù)中的特征
            [peripheral discoverCharacteristics:@[characteristicUUID] forService:service];
        }
    }
}
//外圍設(shè)備尋找到特征后
-(void)peripheral:(CBPeripheral *)peripheral didDiscoverCharacteristicsForService:(CBService *)service error:(NSError *)error{
    NSLog(@"已發(fā)現(xiàn)可用特征...");
    [self writeToLog:@"已發(fā)現(xiàn)可用特征..."];
    if (error) {
        NSLog(@"外圍設(shè)備尋找特征過程中發(fā)生錯誤,錯誤信息:%@",error.localizedDescription);
        [self writeToLog:[NSString stringWithFormat:@"外圍設(shè)備尋找特征過程中發(fā)生錯誤,錯誤信息:%@",error.localizedDescription]];
    }
    //遍歷服務(wù)中的特征
    CBUUID *serviceUUID=[CBUUID UUIDWithString:kServiceUUID];
    CBUUID *characteristicUUID=[CBUUID UUIDWithString:kCharacteristicUUID];
    if ([service.UUID isEqual:serviceUUID]) {
        for (CBCharacteristic *characteristic in service.characteristics) {
            if ([characteristic.UUID isEqual:characteristicUUID]) {
                //情景一:通知
                /*找到特征后設(shè)置外圍設(shè)備為已通知狀態(tài)(訂閱特征):
                 *1.調(diào)用此方法會觸發(fā)代理方法:-(void)peripheral:(CBPeripheral *)peripheral didUpdateNotificationStateForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error
                 *2.調(diào)用此方法會觸發(fā)外圍設(shè)備的訂閱代理方法
                 */
                [peripheral setNotifyValue:YES forCharacteristic:characteristic];
                //情景二:讀取
//                [peripheral readValueForCharacteristic:characteristic];
//                    if(characteristic.value){
//                    NSString *value=[[NSString alloc]initWithData:characteristic.value encoding:NSUTF8StringEncoding];
//                    NSLog(@"讀取到特征值:%@",value);
//                }
            }
        }
    }
}
//特征值被更新后
-(void)peripheral:(CBPeripheral *)peripheral didUpdateNotificationStateForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error{
    NSLog(@"收到特征更新通知...");
    [self writeToLog:@"收到特征更新通知..."];
    if (error) {
        NSLog(@"更新通知狀態(tài)時發(fā)生錯誤,錯誤信息:%@",error.localizedDescription);
    }
    //給特征值設(shè)置新的值
    CBUUID *characteristicUUID=[CBUUID UUIDWithString:kCharacteristicUUID];
    if ([characteristic.UUID isEqual:characteristicUUID]) {
        if (characteristic.isNotifying) {
            if (characteristic.properties==CBCharacteristicPropertyNotify) {
                NSLog(@"已訂閱特征通知.");
                [self writeToLog:@"已訂閱特征通知."];
                return;
            }else if (characteristic.properties ==CBCharacteristicPropertyRead) {
                //從外圍設(shè)備讀取新值,調(diào)用此方法會觸發(fā)代理方法:-(void)peripheral:(CBPeripheral *)peripheral didUpdateValueForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error
                [peripheral readValueForCharacteristic:characteristic];
            }

        }else{
            NSLog(@"停止已停止.");
            [self writeToLog:@"停止已停止."];
            //取消連接
            [self.centralManager cancelPeripheralConnection:peripheral];
        }
    }
}
//更新特征值后(調(diào)用readValueForCharacteristic:方法或者外圍設(shè)備在訂閱后更新特征值都會調(diào)用此代理方法)
-(void)peripheral:(CBPeripheral *)peripheral didUpdateValueForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error{
    if (error) {
        NSLog(@"更新特征值時發(fā)生錯誤,錯誤信息:%@",error.localizedDescription);
        [self writeToLog:[NSString stringWithFormat:@"更新特征值時發(fā)生錯誤,錯誤信息:%@",error.localizedDescription]];
        return;
    }
    if (characteristic.value) {
        NSString *value=[[NSString alloc]initWithData:characteristic.value encoding:NSUTF8StringEncoding];
        NSLog(@"讀取到特征值:%@",value);
        [self writeToLog:[NSString stringWithFormat:@"讀取到特征值:%@",value]];
    }else{
        NSLog(@"未發(fā)現(xiàn)特征值.");
        [self writeToLog:@"未發(fā)現(xiàn)特征值."];
    }
}

#pragma mark - 屬性
-(NSMutableArray *)peripherals{
   if(!_peripherals){
       _peripherals=[NSMutableArray array];
   }
   return _peripherals;
}

#pragma mark - 私有方法
/**
 *  記錄日志
 *
 *  @param info 日志信息
 */
-(void)writeToLog:(NSString *)info{
    self.log.text=[NSString stringWithFormat:@"%@\r\n%@",self.log.text,info];
}

@end

CBPeripheralManager 外設(shè)模式

使用手機作為外設(shè)連接其他中心設(shè)備操作的場景。
PS:因為蘋果設(shè)備的安全性和封閉性,蘋果設(shè)備不能通過與其他設(shè)備藍牙鏈接進行文件傳輸?shù)裙δ?所以在iOS與藍牙開發(fā)的編程中是CBCentralMannager 中心模式編程居多.

1 建立外設(shè)角色
2 設(shè)置本地外設(shè)的服務(wù)和特征
3 發(fā)布外設(shè)和特征
4 廣播服務(wù)
5 響應中心的讀寫請求
6 發(fā)送更新的特征值,訂閱中心

#import "ViewController.h"
#import 
#define kServiceUUID @"C4FB2349-72FE-4CA2-94D6-1F3CB16331EE" //服務(wù)的UUID
#define kCharacteristicUUID @"6A3E4B28-522D-4B3B-82A9-D5E2004534FC" //特征的UUID

@interface ViewController ()

@property (strong,nonatomic) CBCentralManager *centralManager;//中心設(shè)備管理器
@property (strong,nonatomic) NSMutableArray *peripherals;//連接的外圍設(shè)備
@property (weak, nonatomic) IBOutlet UITextView *log;//日志記錄

@end
@implementation ViewController
#pragma mark - 視圖控制器方法
- (void)viewDidLoad {
    [super viewDidLoad];
}

#pragma mark - UI事件
//創(chuàng)建外圍設(shè)備
- (IBAction)startClick:(UIBarButtonItem *)sender {
    _peripheralManager=[[CBPeripheralManager alloc]initWithDelegate:self queue:nil];
}
//更新數(shù)據(jù)
- (IBAction)transferClick:(UIBarButtonItem *)sender {
    [self updateCharacteristicValue];
}

#pragma mark - CBPeripheralManager代理方法
//外圍設(shè)備狀態(tài)發(fā)生變化后調(diào)用
-(void)peripheralManagerDidUpdateState:(CBPeripheralManager *)peripheral{
    switch (peripheral.state) {
        case CBPeripheralManagerStatePoweredOn:
            NSLog(@"BLE已打開.");
            [self writeToLog:@"BLE已打開."];
            //添加服務(wù)
            [self setupService];
            break;

        default:
            NSLog(@"此設(shè)備不支持BLE或未打開藍牙功能,無法作為外圍設(shè)備.");
            [self writeToLog:@"此設(shè)備不支持BLE或未打開藍牙功能,無法作為外圍設(shè)備."];
            break;
    }
}
//外圍設(shè)備添加服務(wù)后調(diào)用
-(void)peripheralManager:(CBPeripheralManager *)peripheral didAddService:(CBService *)service error:(NSError *)error{
    if (error) {
        NSLog(@"向外圍設(shè)備添加服務(wù)失敗,錯誤詳情:%@",error.localizedDescription);
        [self writeToLog:[NSString stringWithFormat:@"向外圍設(shè)備添加服務(wù)失敗,錯誤詳情:%@",error.localizedDescription]];
        return;
    }

    //添加服務(wù)后開始廣播
    NSDictionary *dic=@{CBAdvertisementDataLocalNameKey:kPeripheralName};//廣播設(shè)置
    [self.peripheralManager startAdvertising:dic];//開始廣播
    NSLog(@"向外圍設(shè)備添加了服務(wù)并開始廣播...");
    [self writeToLog:@"向外圍設(shè)備添加了服務(wù)并開始廣播..."];
}
-(void)peripheralManagerDidStartAdvertising:(CBPeripheralManager *)peripheral error:(NSError *)error{
    if (error) {
        NSLog(@"啟動廣播過程中發(fā)生錯誤,錯誤信息:%@",error.localizedDescription);
        [self writeToLog:[NSString stringWithFormat:@"啟動廣播過程中發(fā)生錯誤,錯誤信息:%@",error.localizedDescription]];
        return;
    }
    NSLog(@"啟動廣播...");
    [self writeToLog:@"啟動廣播..."];
}
//訂閱特征
-(void)peripheralManager:(CBPeripheralManager *)peripheral central:(CBCentral *)central didSubscribeToCharacteristic:(CBCharacteristic *)characteristic{
    NSLog(@"中心設(shè)備:%@ 已訂閱特征:%@.",central,characteristic);
    [self writeToLog:[NSString stringWithFormat:@"中心設(shè)備:%@ 已訂閱特征:%@.",central.identifier.UUIDString,characteristic.UUID]];
    //發(fā)現(xiàn)中心設(shè)備并存儲
    if (![self.centralM containsObject:central]) {
        [self.centralM addObject:central];
    }
    /*中心設(shè)備訂閱成功后外圍設(shè)備可以更新特征值發(fā)送到中心設(shè)備,一旦更新特征值將會觸發(fā)中心設(shè)備的代理方法:
     -(void)peripheral:(CBPeripheral *)peripheral didUpdateValueForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error
     */

//    [self updateCharacteristicValue];
}
//取消訂閱特征
-(void)peripheralManager:(CBPeripheralManager *)peripheral central:(CBCentral *)central didUnsubscribeFromCharacteristic:(CBCharacteristic *)characteristic{
    NSLog(@"didUnsubscribeFromCharacteristic");
}
-(void)peripheralManager:(CBPeripheralManager *)peripheral didReceiveWriteRequests:(CBATTRequest *)request{
    NSLog(@"didReceiveWriteRequests");
}
-(void)peripheralManager:(CBPeripheralManager *)peripheral willRestoreState:(NSDictionary *)dict{
    NSLog(@"willRestoreState");
}
#pragma mark -屬性
-(NSMutableArray *)centralM{
    if (!_centralM) {
        _centralM=[NSMutableArray array];
    }
    return _centralM;
}

#pragma mark - 私有方法
//創(chuàng)建特征、服務(wù)并添加服務(wù)到外圍設(shè)備
-(void)setupService{
    /*1.創(chuàng)建特征*/
    //創(chuàng)建特征的UUID對象
    CBUUID *characteristicUUID=[CBUUID UUIDWithString:kCharacteristicUUID];
    //特征值
//    NSString *valueStr=kPeripheralName;
//    NSData *value=[valueStr dataUsingEncoding:NSUTF8StringEncoding];
    //創(chuàng)建特征
    /** 參數(shù)
     * uuid:特征標識
     * properties:特征的屬性,例如:可通知、可寫、可讀等
     * value:特征值
     * permissions:特征的權(quán)限
     */
    CBMutableCharacteristic *characteristicM=[[CBMutableCharacteristic alloc]initWithType:characteristicUUID properties:CBCharacteristicPropertyNotify value:nil permissions:CBAttributePermissionsReadable];
    self.characteristicM=characteristicM;
//    CBMutableCharacteristic *characteristicM=[[CBMutableCharacteristic alloc]initWithType:characteristicUUID properties:CBCharacteristicPropertyRead value:nil permissions:CBAttributePermissionsReadable];
//    characteristicM.value=value;

    /*創(chuàng)建服務(wù)并且設(shè)置特征*/
    //創(chuàng)建服務(wù)UUID對象
    CBUUID *serviceUUID=[CBUUID UUIDWithString:kServiceUUID];
    //創(chuàng)建服務(wù)
    CBMutableService *serviceM=[[CBMutableService alloc]initWithType:serviceUUID primary:YES];
    //設(shè)置服務(wù)的特征
    [serviceM setCharacteristics:@[characteristicM]];

    /*將服務(wù)添加到外圍設(shè)備*/
    [self.peripheralManager addService:serviceM];
}
//更新特征值
-(void)updateCharacteristicValue{
    //特征值
    NSString *valueStr=[NSString stringWithFormat:@"%@ --%@",kPeripheralName,[NSDate   date]];
    NSData *value=[valueStr dataUsingEncoding:NSUTF8StringEncoding];
    //更新特征值
    [self.peripheralManager updateValue:value forCharacteristic:self.characteristicM onSubscribedCentrals:nil];
    [self writeToLog:[NSString stringWithFormat:@"更新特征值:%@",valueStr]];
}
/**
 *  記錄日志
 *
 *  @param info 日志信息
 */
-(void)writeToLog:(NSString *)info{
    self.log.text=[NSString stringWithFormat:@"%@\r\n%@",self.log.text,info];
}
@end

藍牙連接發(fā)送數(shù)據(jù)的流程圖

16a2962d7ce67281.png

lightblue怎么用

OTA(空中升級)

將手機作為外圍設(shè)備開發(fā)

iOS淺析藍牙設(shè)備之服務(wù)器(外圍設(shè)備)

APP被作為外設(shè)

遇到的坑

截屏2019-10-2315.57.54.png

所以得知道 設(shè)備廣播的的UUID
或者用下面代碼搜索藍牙

 [self.centralManager scanForPeripheralsWithServices:nil options:nil];

一般藍牙連接失敗我們會實現(xiàn)以下方法

// 連接失敗(但不包含超時,系統(tǒng)沒有超時處理)
- (void)centralManager:(CBCentralManager *)central didFailToConnectPeripheral:(CBPeripheral *)peripheral error:(NSError *)error {
    if (error && self.delegate && [self.delegate respondsToSelector:@selector(centralTool:connectFailure:)]) {
        [self.delegate centralTool:self connectFailure:error];
    }
}

- (void)centralManager:(CBCentralManager *)central didDisconnectPeripheral:(CBPeripheral *)peripheral error:(NSError *)error
{
    [self autoConnect];
}

但是當廣播設(shè)備的藍牙斷開再想重連的話還要實現(xiàn)下面這個方法

- (void)peripheral:(CBPeripheral *)peripheral didModifyServices:(NSArray<CBService *> *)invalidatedServices{

    [self.centralManager connectPeripheral:peripheral options:nil];
      // 注意保留 Peripheral 的引用
    self.lastConnectedPeripheral = peripheral;
    [self startTimer];
}

只有這樣才能實現(xiàn)斷開藍牙重開藍牙秒連

另外,有時候要提高搜索效率的話,可以過濾藍牙的名字

    NSString *name = peripheral.name;
       if (name.length == 0) {
           return;
       }
       NSLog(@"name ===  %@",name);
       
       NSString *string = [NSString stringWithFormat:@"已發(fā)現(xiàn) peripheral: %@ rssi: %@, UUID: %@ advertisementData: %@ ", peripheral, RSSI, peripheral.identifier, advertisementData];
       NSLog(@"string ==  %@",string);
    


       if ([name rangeOfString:@"Berry"].location == NSNotFound) {
           return;
       }
    
   
//    NSString *kCBAdvDataLocalName = [advertisementData objectForKey:CBAdvertisementDataLocalNameKey];
//    if ([kCBAdvDataLocalName rangeOfString:@"Berry"].location == NSNotFound) {
//             return;
//         }

不過要注意 上文代碼中的name 和 kCBAdvDataLocalName 是不同的
一個是藍牙設(shè)備的名字, 一個是廣播的名字

//添加服務(wù)后開始廣播
//廣播設(shè)置
    NSDictionary *dic=@{CBAdvertisementDataLocalNameKey:MyDeviceName,CBAdvertisementDataServiceUUIDsKey:@[[CBUUID UUIDWithString:@"FFE0"]]};
//廣播設(shè)置
    [self.peripheralManager startAdvertising:dic];//開始廣播

第一個坑中設(shè)置UUID找不到的問題就是,需要在上述代碼中設(shè)置UUID
網(wǎng)上找的大部分資料都是關(guān)于如何設(shè)置中心設(shè)備的,外設(shè)的比較少,所以這里容易出錯

不過,如果這塊都設(shè)置了,對于快速精準找到目標藍牙服務(wù)是大有裨益的,也會減少耗電量,提高響應速度等等。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容