基于iOS的蓝牙开发

导语:

最近几天在做一个关于蓝牙与血压计和血氧仪交互方面的东西,刚开始使用的是babybluetooth(但封装的内容我不会使用,于是就自己写了一个工具类)。在此带来一些关于蓝牙开发的分享。

1.蓝牙基础知识

CoreBluetooth框架的核心是peripheral(外设)和central(中心),发起连接的是 central,被连接的设备为 peripheral。在移动端开发中,我们通常使用的是中心模式。

2.中心模式的流程

  1. 建立中心
  2. 扫描外设(discover)
  3. 连接外设(connect)
  4. 扫描外设中的服务和特征(discover)
- 4.1 获取外设的 services
- 4.2 获取外设的 Characteristics,获取Characteristics的值,获 Characteristics的 Descriptor 和 Descriptor 的值
  1. 与外设做数据交互(explore and interact)
  2. 订阅 Characteristic 的通知
  3. 断开连接(disconnect)

3.具体实现代码

创建中心管理者

#import <CoreBluetooth/CoreBluetooth.h>
typedef NS_ENUM(NSInteger,BDBlueToothType){
    BDBlueToothType_Oximeter = 1,//血氧仪
    BDBlueToothType_Hamnatodynamometer //血压计
};
//蓝牙搜索到的设备数组
typedef void(^PeripheralBlock)(NSMutableArray *);
//读到的数据
typedef void(^ReadValueBlock)(NSString *);
@interface BDBlueToothHelper : NSObject<CBCentralManagerDelegate, CBPeripheralDelegate>
//外设
@property(nonatomic, strong) CBPeripheral* myPeripheral;
//中心管理工具
@property (nonatomic, strong) CBCentralManager* myCentralManager;

初始化开始扫描

self.myCentralManager = [[CBCentralManager alloc]initWithDelegate:self queue:nil options:nil];
必须实现的代理方法
//查看蓝牙服务
- (void)centralManagerDidUpdateState:(CBCentralManager *)central{
    switch (central.state) {
        case CBCentralManagerStatePoweredOn:
            BDLog(@"蓝牙已打开, 请扫描外设!");
            //搜索外设
            [self.myCentralManager scanForPeripheralsWithServices:nil options:@{CBCentralManagerOptionShowPowerAlertKey:@YES}];
            break;
        case CBCentralManagerStatePoweredOff:
            BDLog(@"蓝牙关闭...");
            break;
        default:
            break;
    }
}

#pragma mark 搜索到设备之后会调用代理方法
- (void)centralManager:(CBCentralManager *)central // 中心管理者
 didDiscoverPeripheral:(CBPeripheral *)peripheral // 外设
     advertisementData:(NSDictionary *)advertisementData // 外设携带的数据
                  RSSI:(NSNumber *)RSSI{ // 外设发出的蓝牙信号强度
    BDLog(@"已发现 peripheral: %@ rssi: %@, name: %@ advertisementData: %@", peripheral, RSSI, peripheral.name, advertisementData);
    //这里可以做一些过滤操作
    if ([_myPeripherals containsObject:peripheral]) {
        
    }else
    {
        //找到的设备必须持有它,否则CBCentralManager中也不会保存peripheral
        [_myPeripherals addObject:peripheral];
    }
   //将搜索的设备回调给控制器的tableview使用,刷新表格
    if (self.perlists) {
        self.perlists(_myPeripherals);
    }
    
}

建立连接

//self.myPeripheral在cell点击时的赋值
[self.myCentralManager connectPeripheral:self.myPeripheral options:nil];

一个主设备最多能连7个外设,每个外设最多只能给一个主设备连接,连接成功、失败、断开都会进入到相应的代理

#pragma mark 连接外设成功
- (void)centralManager:(CBCentralManager *)central didConnectPeripheral:(CBPeripheral *)peripheral{
    BDLog(@"成功外设连接");
    //设置外设的代理
    [self.myPeripheral setDelegate:self];
    //外设发现服务,传nil代表不过滤
    // 这里会触发外设的代理方法 - (void)peripheral:(CBPeripheral *)peripheral didDiscoverServices:(NSError *)error
    [self.myPeripheral discoverServices:nil];
}

#pragma mark 掉线时调用 丢失连接
- (void)centralManager:(CBCentralManager *)central didDisconnectPeripheral:(CBPeripheral *)peripheral error:(NSError *)error{
    UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"提示" message:@"蓝牙连接已断开" delegate:nil cancelButtonTitle:nil otherButtonTitles:@"关闭", nil];
    [alert show];
    [self closeConnect];
    BDLog(@"丢失连接");
}

#pragma mark 连接外设失败
- (void)centralManager:(CBCentralManager *)central didFailToConnectPeripheral:(CBPeripheral *)peripheral error:(NSError *)error{
    BDLog(@"连接外设失败%@", error);
}

扫描服务和特征

#pragma mark 发现服务
- (void)peripheral:(CBPeripheral *)peripheral didDiscoverServices:(NSError *)error{
    BDLog(@"发现服务!");
    for(CBService* s in peripheral.services){
        NSLog(@"%d :服务 UUID: %@(%@)", i, s.UUID.data, s.UUID);
        //扫描每个service的Characteristics,扫描到后会进入方法: -(void)peripheral:(CBPeripheral *)peripheral didDiscoverCharacteristicsForService:(CBService *)service error:(NSError *)error
        [peripheral discoverCharacteristics:nil forService:s];
        [self.nServices addObject:s];
    }
}

#pragma mark 发现外设服务里的特征的时候调用的代理方法
- (void)peripheral:(CBPeripheral *)peripheral didDiscoverCharacteristicsForService:(CBService *)service error:(NSError *)error{
    for(CBCharacteristic* c in service.characteristics){
        BDLog(@"特征 UUID: %@ (%@)", c.UUID.data, c.UUID);
        
        if (self.toothType == BDBlueToothType_Oximeter) {//血氧仪
            if([c.UUID isEqual:[CBUUID UUIDWithString:@"FFF1"]]){
                self.writeCharacteristic = c;
                
                BDLog(@"找到WRITE : %@", c);
            }else if([c.UUID isEqual:[CBUUID UUIDWithString:@"FFF4"]]){
                self.readCharacteristic = c;
                
                [self.myPeripheral setNotifyValue:YES forCharacteristic:c];
                [self.myPeripheral readValueForCharacteristic:c];
                NSLog(@"找到READ : %@", c);
            }
        }else if (self.toothType == BDBlueToothType_Hamnatodynamometer){//血压计
            if ([c.UUID isEqual:[CBUUID UUIDWithString:@"FFE1"]]) {
                BDLog(@"血压计 %zd - %@",c.properties,c.descriptors);
                self.readCharacteristic = c;
                self.writeCharacteristic = c;
                [self.myPeripheral setNotifyValue:YES forCharacteristic:c];
                [self.myPeripheral readValueForCharacteristic:c];
            }
        }
        
    }
}

与外设进行数据交互

#pragma mark 获取外设发来的数据,不论是read和notify,获取数据都从这个方法中读取
- (void)peripheral:(CBPeripheral *)peripheral didUpdateValueForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error{
    [peripheral readRSSI];
    
    NSData* data = characteristic.value;
    
    if (!data.length) {
        return;
    }
    
    NSString* value = [self hexadecimalString:data];
    
    if (self.toothType == BDBlueToothType_Oximeter) { //血氧仪
        
        if([characteristic.UUID isEqual:[CBUUID UUIDWithString:@"FFF4"]]){
            if (value.length) {
                self.readValue(value);
            }
//            BDLog(@"characteristic : %@, data : %@, value : %@", characteristic, data, value);
        }

    }else if (self.toothType == BDBlueToothType_Hamnatodynamometer){ //血压计
        
        if ([characteristic.UUID isEqual:[CBUUID UUIDWithString:@"FFE1"]]) {
            if (value.length) {
                self.readValue(value);
            }
//            BDLog(@"characteristic : %@, data : %@, value : %@", characteristic, data, value);
        }
    }
}

//中心读取外设实时数据
- (void)peripheral:(CBPeripheral *)peripheral didUpdateNotificationStateForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error{
    if(error){
        BDLog(@"Error changing notification state: %@", error.localizedDescription);
    }
    
    if(characteristic.isNotifying){
        [peripheral readValueForCharacteristic:characteristic];
    }else{
        BDLog(@"Notification stopped on %@. Disconnting", characteristic);
        [self.myCentralManager cancelPeripheralConnection:self.myPeripheral];
    }
}

//向peripheral中写入数据
- (void)writeToPeripheral:(NSString *)data{
    if(!_writeCharacteristic){
        BDLog(@"writeCharacteristic is nil!");
        return;
    }
    NSData* value = [self dataWithHexstring:data];
    
    [_myPeripheral writeValue:value forCharacteristic:_writeCharacteristic type:CBCharacteristicWriteWithResponse];
}

//向peripheral中写入数据后的回调函数
- (void)peripheral:(CBPeripheral*)peripheral didWriteValueForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error{
    if (error) {
        BDLog(@"error %@",error);
    }
    BDLog(@"write value success : %@", characteristic);
}

用到的私有方法

//将传入的NSData类型转换成NSString并返回
- (NSString*)hexadecimalString:(NSData *)data{
    NSString* result;
    const unsigned char* dataBuffer = (const unsigned char*)[data bytes];
    if(!dataBuffer){
        return nil;
    }
    NSUInteger dataLength = [data length];
    NSMutableString* hexString = [NSMutableString stringWithCapacity:(dataLength * 2)];
    for(int i = 0; i < dataLength; i++){
        [hexString appendString:[NSString stringWithFormat:@"%02lx", (unsigned long)dataBuffer[i]]];
    }
    result = [NSString stringWithString:hexString];
    return result;
}

//将传入的NSString类型转换成NSData并返回
- (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;
}

相关的蓝牙文档

屏幕快照 2017-07-24 上午11.43.21.png

参考文档

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

推荐阅读更多精彩内容

  • 这里我们具体说明一下中心模式的应用场景。主设备(手机去扫描连接外设,发现外设服务和属性,操作服务和属性的应用。一般...
    丶逝水流年阅读 2,250评论 3 4
  • iOS连接外设的代码实现流程 1. 建立中心角色 2. 扫描外设(discover) 3. 连接外设(connec...
    UILabelkell阅读 2,413评论 2 4
  • 本文主要以蓝牙4.0做介绍,因为现在iOS能用的蓝牙也就是只仅仅4.0的设备 用的库就是core bluetoot...
    暮雨飞烟阅读 836评论 0 2
  • 1.smarty的增删改差 2.Smarty缓存 提高网站的运行效率,减轻数据库的压力,针对的数据是非实时更新数据...
    AnthonyHu阅读 92评论 0 0
  • 2016年啦,只是希望通过这篇文章一直延续时不时来简书写点小随笔、日记的开端,希望这个空间能一直被经营下去。...
    惟度阅读 507评论 1 2