蓝牙通信

BLE中心模式流程-coding

BLE中心模式流程

  • 1.建立中心角色
  • 2.扫描外设(Discover Peripheral)
  • 3.连接外设(Connect Peripheral)
  • 4.扫描外设中的服务和特征(Discover Services And Characteristics)
    • 4.1 获取外设的services
    • 4.2 获取外设的Characteristics,获取characteristics的值,,获取Characteristics的Descriptor和Descriptor的值
  • 5.利用特征与外设做数据交互(Explore And Interact)
  • 6.订阅Characteristic的通知
  • 7.断开连接(Disconnect)

准备环境

  • 1.Xcode
  • 2.下载蓝牙调试助手
  • 3.外设

实现步骤

#define UUID0 @"FFE0"      // 外设服务
#define UUID1 @"FFE1"       // 外设特征值(write)
#define UUID2 @"FFF2"       // 外设特征值(read)
#define UUID3 @"FFF3"       // 外设特征值(notify)
#define UUID4 @"FFF4"      // 外设特征值(扫描过滤)

导入CB头文件,建立主设备管理类,设置主设备代理

#import <CoreBluetooth/CoreBluetooth.h>
@property (nonatomic, strong) CBCentralManager *cMgr; /**< 中心管理设备 */
- (CBCentralManager *)cMgr {
    if (!_cMgr) {
        _cMgr = [[CBCentralManager alloc] initWithDelegate:self queue:dispatch_get_main_queue()]; // 线程不传默认是主线程
    }
    return _cMgr;
}
  1. 设备蓝牙状态更新回调
#pragma mark 蓝牙状态更新回调
- (void)centralManagerDidUpdateState:(CBCentralManager *)central {
    LogFunc
    self.peripheralState = central.state;
    switch (central.state) {
        case CBManagerStateUnknown: {
            NSLog(@"为知状态");
        }
            break;
        case CBManagerStateResetting: {
            NSLog(@"重置状态");
        }
            break;
        case CBManagerStateUnsupported: {
            NSLog(@"不支持的状态");
        }
            break;
        case CBManagerStateUnauthorized: {
            NSLog(@"未授权的状态");
        }
            break;
        case CBManagerStatePoweredOff: {
            NSLog(@"关闭状态");
            if ([self.delegate respondsToSelector:@selector(bluetoothTool:didDisconnectPeripheral:error:)]) {
                [self.delegate bluetoothTool:self didDisconnectPeripheral:nil error:nil];
            }
            self.peripheral = nil;
            self.writeCharacteristic = nil;
            self.notifyCharacteristic = nil;
            self.readCharacteristic = nil;
        }
            break;
        case CBManagerStatePoweredOn: {
            NSLog(@"开启状态-可用状态");
            self.peripheralState = central.state;
            if ([self.delegate respondsToSelector:@selector(powerOn:)]) {
                [self.delegate powerOn:self];
            }
        }
            break;
        default:
            break;
    }
}
  1. 开始扫描:
// 传nil会扫描所有设备,可以根据需求传入你们硬件给的过滤的UDID
[self.centralManager scanForPeripheralsWithServices:nil options:nil];
  1. 发现外设
#pragma mark 发现外设
- (void)centralManager:(CBCentralManager *)central didDiscoverPeripheral:(CBPeripheral *)peripheral advertisementData:(NSDictionary<NSString *,id> *)advertisementData RSSI:(NSNumber *)RSSI {
 // 连接外设,获取广播数据,比如mac地址等等
        self.peripheral = peripheral;
        [self.centralManager connectPeripheral:peripheral options:nil];
        NSData *adata = advertisementData[@"kCBAdvDataManufacturerData"];
        NSMutableString *string = [[NSMutableString alloc] initWithCapacity:[adata length]];
        [adata enumerateByteRangesUsingBlock:^(const void *bytes, NSRange byteRange, BOOL *stop) {
            unsigned char *dataBytes = (unsigned char*)bytes;
            for (NSInteger i = 0; i < byteRange.length; i++) {
                NSString *hexStr = [NSString stringWithFormat:@"%x", (dataBytes[i]) & 0xff];
                if ([hexStr length] == 2) {
                    [string appendString:hexStr];
                } else {
                    [string appendFormat:@"0%@", hexStr];
                }
            }
        }];
    }
}

连接成功

#pragma mark 连接成功
- (void)centralManager:(CBCentralManager *)central didConnectPeripheral:(CBPeripheral *)peripheral {
    LogFunc
    NSLog(@"连接成功");
    // 停止扫描
[self.centralManager stopScan];
    self.peripheral = peripheral;
    self.peripheral.delegate = self;
    CBUUID *uuid = [CBUUID UUIDWithString:UUID0];
    NSArray *serives = @[uuid];
    [peripheral discoverServices:serives];
}
#pragma mark 连接失败
- (void)centralManager:(CBCentralManager *)central didFailToConnectPeripheral:(CBPeripheral *)peripheral error:(NSError *)error {
    LogFunc;
    NSLog(@"连接失败 %@", error);
}
#pragma mark 连接断开
- (void)centralManager:(CBCentralManager *)central didDisconnectPeripheral:(CBPeripheral *)peripheral error:(NSError *)error {
    LogFunc
    NSLog(@"连接断开了 %@", error);
    self.peripheral = nil;
    self.writeCharacteristic = nil;
    self.notifyCharacteristic = nil;
}

3.发现服务

#pragma mark 发现服务
- (void)peripheral:(CBPeripheral *)peripheral didDiscoverServices:(NSError *)error {
    LogFunc
    // 遍历所有的服务
    for (CBService *service in peripheral.services) {
        if ([service.UUID.UUIDString isEqualToString:UUID0]) {
            self.cbService = service;
            [self.peripheral discoverCharacteristics:nil forService:service];
            break;
        }
    }
}
  1. 发现特征值
#pragma mark 发现特征值
- (void)peripheral:(CBPeripheral *)peripheral didDiscoverCharacteristicsForService:(CBService *)service error:(NSError *)error {
    LogFunc
  for (CBCharacteristic *characteristic in service.characteristics) {
        
//        NSLog(@">>>服务:%@ 的 特征: %@,特征属性properties:%d",service.UUID.UUIDString, characteristic.UUID.UUIDString, characteristic.properties);
        if ([characteristic.UUID.UUIDString isEqualToString:WRITE_UUID]) {
            self.writeCharacteristic = characteristic;
        } else if ([characteristic.UUID.UUIDString isEqualToString:NOTIFY_UUID]) {
            self.notifyCharacteristic = characteristic;
                //订阅,实时接收
            if (self.notifyCharacteristic.properties & CBCharacteristicPropertyNotify) {
                [self.peripheral setNotifyValue:YES forCharacteristic:self.notifyCharacteristic];
//                [self.peripheral readValueForCharacteristic:self.notifyCharacteristic];
            }
        } else if ([characteristic.UUID.UUIDString isEqualToString:READ_UUID]) {
            self.readCharacteristic = characteristic;
            if (self.readCharacteristic.properties & CBCharacteristicPropertyRead) {
                [self.peripheral readValueForCharacteristic:self.readCharacteristic];
            }
        }
    }
// 测试写数据
  // 蓝牙重置测试
    NSString *resetString = @"00000302";
    NSData *resetData = [Tool dataForHexString:resetString];
    
//                NSData *testData = [[NSData alloc]initWithBytes:byteArray length:6];
//                NSString *hexStr = [Tool ConvertToNSString:netData];
    if (self.writeCharacteristic.properties & CBCharacteristicPropertyWrite) {
        [self.peripheral writeValue:frequencyData forCharacteristic:self.writeCharacteristic type:CBCharacteristicWriteWithoutResponse];
    }
}
  1. 发现描述descrptor 根据需求添加

  2. 接收蓝牙数据

#pragma mark 接收蓝牙数据
- (void)peripheral:(CBPeripheral *)peripheral didUpdateValueForCharacteristic:(nonnull CBCharacteristic *)characteristic error:(nullable NSError *)error {
    LogFunc;
    if (error) {
        NSLog(@"error: %@", error);
        return;
    }
    NSData *data = characteristic.value;
    NSLog(@"uuid:%@ data:%@",characteristic.UUID.UUIDString, data);
}

  1. 蓝牙写入数据回调
#pragma mark 蓝牙写入数据回调  type为CBCharacteristicWriteWithResponse回调用
- (void)peripheral:(CBPeripheral *)peripheral didWriteValueForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error {
    LogFunc;
    if (error) {
    } 
}

通知回调

#pragma mark 设置通知回调
- (void)peripheral:(CBPeripheral *)peripheral didUpdateNotificationStateForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error {
    LogFunc
    if (error) {
        NSLog(@"error:%@", error);
        return;
    }
    if (characteristic.isNotifying) {
        [peripheral readValueForCharacteristic:characteristic];
    }
}

8.项目中遇到的坑

  • 写给外设数据格式问题
    • 跟硬件工程师沟通,需要传输的数据类型,一般都是传byte字节数组,可能需要用到的方法
#pragma mark - MD5加密
-(NSString *)md5:(NSString *)inPutText
{
    const char *cStr = [inPutText UTF8String];
    unsigned char result[CC_MD5_DIGEST_LENGTH];
    CC_MD5(cStr, strlen(cStr), result);
    
    return [[NSString stringWithFormat:@"%02X%02X%02X%02X%02X%02X%02X%02X%02X%02X%02X%02X%02X%02X%02X%02X",
             result[0], result[1], result[2], result[3],
             result[4], result[5], result[6], result[7],
             result[8], result[9], result[10], result[11],
             result[12], result[13], result[14], result[15]
             ] lowercaseString];
}

#pragma mark - 将NSString转换成十六进制的字符串则可使用如下方式:
- (NSString *)convertStringToHexStr:(NSString *)str {
    if (!str || [str length] == 0) {
        return @"";
    }
    NSData *data = [str dataUsingEncoding:NSUTF8StringEncoding];
    
    NSMutableString *string = [[NSMutableString alloc] initWithCapacity:[data length]];
    
    [data enumerateByteRangesUsingBlock:^(const void *bytes, NSRange byteRange, BOOL *stop) {
        unsigned char *dataBytes = (unsigned char*)bytes;
        for (NSInteger i = 0; i < byteRange.length; i++) {
            NSString *hexStr = [NSString stringWithFormat:@"%x", (dataBytes[i]) & 0xff];
            if ([hexStr length] == 2) {
                [string appendString:hexStr];
            } else {
                [string appendFormat:@"0%@", hexStr];
            }
        }
    }];
    
    return string;
}
#pragma mark - 将十六进制的字符串转换为需要的data,这就是你最终需要写入的数据:
+ (NSData *)dataWithHexString:(NSString *)hexstring
{
    NSMutableData* data = [NSMutableData data];
    int idx;
    for (idx = 0; idx+2 <= hexstring.length; idx+=2) {
        NSRange range = NSMakeRange(idx, 2);
        NSString* hexStr = [hexstring substringWithRange:range];
        NSScanner* scanner = [NSScanner scannerWithString:hexStr];
        unsigned int intValue;
        [scanner scanHexInt:&intValue];
        [data appendBytes:&intValue length:1];
    }
    return data;
}
  • 读取数据格式问题
    • 读取的也是byte数组
#pragma mark - 更新特征的value的时候会调用 (凡是从蓝牙传过来的数据都要经过这个回调,简单的说这个方法就是你拿数据的唯一方法) 你可以判断是否
- (void)peripheral:(CBPeripheral *)peripheral didUpdateValueForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error
{
     NSLog(@"%s, line = %d", __FUNCTION__, __LINE__);
    NSLog(@"***********");
    if (error) {
        NSLog(@"error:%@",error.localizedDescription);
        return;
    }
    NSData * data = characteristic.value;
    Byte * resultByte = (Byte *)[data bytes];
    NSLog(@"rusultByte:%s",resultByte);
    for(int i=0;i<[data length];i++){
        printf("testByteFF02[%d] = %d\n",i,resultByte[i]);
        if (resultByte[i]==0) { // 授权成功
            
            self.count++;
            self.writeFinishFlag = YES;
            break;
            
           // [self.deviceManager writeValue:[@"Lock" dataUsingEncoding:NSUTF8StringEncoding] forCharacteristic:characteristic type:CBCharacteristicWriteWithResponse];
        }else{
            self.writeFinishFlag = NO;
        }
    }
    if (self.count==1&& self.writeFinishFlag) { // // 授权成功
        NSLog(@"授权成功");
        if (self.lockCharacteristic.properties & CBCharacteristicPropertyWrite) {
         //   [self notifyCharacteristic:peripheral characteristic:characteristic];
           // NSData *data = [NSData dataWithHexString:@"Lock"];
          NSString *lock = [self convertStringToHexStr:@"Lock"];
            NSLog(@"lock:%@",lock);
           // NSData *data = [@"Lock" dataUsingEncoding:NSUTF8StringEncoding];
            NSData *data = [NSData dataWithHexString:lock];
            NSLog(@"lockData:%@",data);
            [self.deviceManager writeValue:data forCharacteristic:self.lockCharacteristic type:CBCharacteristicWriteWithResponse];// 激活锁
            if (characteristic.properties & CBCharacteristicPropertyRead) {
               // [peripheral readValueForCharacteristic:characteristic];
            }
            
        }
    }
    if (self.count==2&& self.writeFinishFlag) {// 激活成功
       NSLog(@"激活成功");
        if (self.lockCharacteristic.properties & CBCharacteristicPropertyWrite) {
           // [self notifyCharacteristic:peripheral characteristic:characteristic];
            NSString *unLock = [self convertStringToHexStr:@"Unlock"];
            NSData *data = [NSData dataWithHexString:unLock];
            NSLog(@"unlockData;%@",data);
            [self.deviceManager writeValue:data forCharacteristic:self.lockCharacteristic type:CBCharacteristicWriteWithResponse];// 开锁
            if (characteristic.properties & CBCharacteristicPropertyRead) {
               // [peripheral readValueForCharacteristic:characteristic];
            }
        }
    }
    if (self.count==3&& self.writeFinishFlag) {// 开锁成功
        NSLog(@"开锁成功");
        [self.bleManager cancelPeripheralConnection:self.deviceManager];// 断开链接
        [self.bleManager stopScan];
        self.bleManager = nil;
    }

}
//base64解密
    NSData *decodedData = [[NSData alloc] initWithBase64EncodedString:base64String options:0];
    NSString *decodedString = [[NSString alloc] initWithData:decodedData encoding:NSUTF8StringEncoding];

// 遇到的问题
蓝牙调试助手测试notify,要先设置notify为true才能监听数据读写
代码也是,要先拿到特征值并设置为true,才能去测试写数据,不然代理方法监听不到数据读入。
报错:error: Error Domain=CBATTErrorDomain Code=2 "Reading is not permitted." UserInfo={NSLocalizedDescription=Reading is not permitted.}
设置notify特征值的时候不要去调入读取数据的方法,[peripheral readValueForCharacteristic:characteristic];不然就会报错。设置notify不需要调入这个方法,只需要设置notify = true就可以了,这个方法是给读取特征值去调入的,notify不需要

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 225,208评论 6 524
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 96,502评论 3 405
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 172,496评论 0 370
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 61,176评论 1 302
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 70,185评论 6 401
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 53,630评论 1 316
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 41,992评论 3 431
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 40,973评论 0 280
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 47,510评论 1 325
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 39,546评论 3 347
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 41,659评论 1 355
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 37,250评论 5 351
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 42,990评论 3 340
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 33,421评论 0 25
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 34,569评论 1 277
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 50,238评论 3 382
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 46,732评论 2 366

推荐阅读更多精彩内容

  • 最近学习蓝牙,所以找了一些资料学习研究了一下蓝牙通讯的一个流程。写了一个小demo,demo效果见下图: 需要de...
    NicoalsNC阅读 1,187评论 0 7
  • 前言: 本文参考文献:1、https://learn.adafruit.com/introduction-to-b...
    esonyf阅读 5,259评论 3 10
  • 原文:http://www.myexception.cn/operating-system/2052286.htm...
    KYM1988阅读 1,962评论 2 2
  • 在iOS开发中,实现蓝牙通信有两种方式,一种是使用传统的GameKit.framework,另一种就是使用在iOS...
    iOS祎阅读 303评论 0 0
  • iOS开发之蓝牙通讯 一、引言 蓝牙是设备近距离通信的一种方便手段,在iPhone引入蓝牙4.0后,设备之间的通讯...
    隐身人阅读 2,401评论 3 0