iOS蓝牙空中升级(固件升级)

空中升级又叫固件升级,指你手机从服务器下载下来的包或者数据,通过蓝牙传输给你的外设升级固件。如果你能把蓝牙的基础搞懂,其实也并不是很难,我在这里只不过提供一下思路。

空中升级略难的地方在于数据处理和交互,尤其要以怎样简单完整的代码来实现数据的读写是重点,这就需要你和硬件工程师的交流和你自己的逻辑思维了。

在上代码以前,说一下有关蓝牙的传输速度的,因为我开发中碰到较大数据的传输,着实害我费了很多脑筋。
蓝牙数据传输中有连接延迟。其是为了低功耗考虑,允许从机在跳频过程中不理会主机的跳频指令,继续睡眠一段时间。而主机不能因为从机睡眠而认为其断开连接了。其是1.25毫秒一个单位。明显,这个数值越小,传输速度也高。
蓝牙BLE协议规定连接参数最小是5,即7.25毫秒;而Android手机规定连接参数最小是8,即10毫秒。iOS规定是16,即20毫秒。
连接参数完全由主机决定,但从机可以发出更新参数申请,主机可以接受也可以拒绝。Android手机一部接受,而ios比较严格,拒绝的概率比较高。
一般场景,连接参数设置16,即20毫秒,一般的传输速率是50* 20 = 1000字节/每秒。如果每个连接事件传输更多的包,可以获得更高的传输速率。
但是以上这种方法并不能真正解决传输的速度快慢,顶多也就相差2倍或者3倍。最好的方法就是在与app每次给蓝牙发送的包数,通畅可能考虑到数据不丢失,都是一包一包的发送,但是在空中升级这里不得已包数必须要多一点,比如一次发送十包,具体还是看你们硬件那边怎么写蓝牙协议了。

我下面的demo是这样的一个过程:
1.发送给外设指令,我要空中升级
->2.外设给我回OK之后我发送一个随机数(自定义了一种随机算法),验证开始固件升级
->3.判断随机数无误,准备发送打包好的数据
->4.真正发送打包好的数据(每次发送10包,一包20个字节),这里会重复N多次,看你的原数据包有多大;每次接到我发的包后,外设都会给我会OK否,我收到OK后才会发一下个数据包
->5.告诉外设我数据发送完毕,并发送一段指令(包括本次空中升级数据包的大小,还有加密参数什么的)
->6.外设给我回OK无误后,才算真正升级完成

//更新特征的value时调用
-(void)peripheral:(CBPeripheral *)peripheral didUpdateValueForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error
{
    if (error) {
        return;
    }
    //找到已经订阅的串口,输出看结果
   if ([[characteristic.UUID UUIDString] isEqualToString:@"6E400003-B5A3-F393-E0A9-E50E24DCCA9E"]) {
       NSLog(@"返回的结果是 = %@",characteristic.value);
       
       [_dataArray addObject:characteristic.value];
       
       NSInteger arrayCount = _dataArray.count;
       //蓝牙每次都会回三条数据
       if (arrayCount%3 == 0) {
           //返回的头
           NSString *str=[[NSString alloc]initWithFormat:@"%@",_dataArray[arrayCount-3]];

           /*第一种大情况
            1.发送固件升级指令
            2.发送随机数
            3.验证随机数是否正确
            */
           if ([str isEqualToString:@"<ab100000 00000000>"]) {
               
               NSData * data2 = _dataArray[arrayCount-1];
               NSString * string3 = [NSString stringWithFormat:@"%@",_dataArray[arrayCount-1]];

               //keyHead
               NSData * keyHead = [_dataArray[arrayCount-1] subdataWithRange:NSMakeRange(2, 1)];
               NSString * keyHeadStr = [NSString stringWithFormat:@"%@",keyHead];
               NSData * randomData1;
               NSData * randomData2;
               //随机数
               if (data2.length == 7 ) {
                   randomData1 = [_dataArray[arrayCount-1] subdataWithRange:NSMakeRange(5, 1)];
                   randomData2 = [_dataArray[arrayCount-1] subdataWithRange:NSMakeRange(6, 1)];
               }
               
               //发起固件升级之后回的
               if ([string3 isEqualToString:@"<01008204 00010000 00>"]) {
                   //写入随机数
                   [self.peripherale writeValue:self.randomData forCharacteristic:self.TX_Characteristic type:CBCharacteristicWriteWithResponse];
                   NSLog(@"写入的随机数 %@",self.randomData);
               }
               
               //写入随机数之后回的
               if ([randomData1 isEqualToData:[_calculateRandom subdataWithRange:NSMakeRange(13, 1)]] && [randomData2 isEqualToData:[_calculateRandom subdataWithRange:NSMakeRange(14, 1)]] && [keyHeadStr isEqualToString:@"<06>"]){
                   
                   //随机数验证成功
                   [self.peripherale writeValue:_successData forCharacteristic:self.TX_Characteristic type:CBCharacteristicWriteWithResponse];
                   NSLog(@"随机数验证成功");
               }
               
               //随机数验证成功之后
               if ([string3 isEqualToString:@"<01000501 0080>"]) {
                   //发送第一包数据包
                   [self.peripherale writeValue:self.packArray[_sendNumber] forCharacteristic:self.TX_Characteristic type:CBCharacteristicWriteWithResponse];
                   NSLog(@"发送的包 %@",self.packArray[_sendNumber]);
                   _sendNumber++;
                   [self setValue:[NSString stringWithFormat:@"%d",_sendNumber] forKey:@"sendNumber"];
               }
           }
           
           /*第二种大情况
            1.校验发送的包是否收到了
            2.取消升级
            */
           else if ([str isEqualToString:@"<ab100000 00001000>"]) {
               
               NSData * data3 = [_dataArray[arrayCount-1] subdataWithRange:NSMakeRange(0, 5)];
               NSString * string3 = [NSString stringWithFormat:@"%@",data3];
               
               if ([string3 isEqualToString:@"<01000804 00>"] && _sendNumber < self.allSection-1) {
                   //发送数据包
                   [self.peripherale writeValue:self.packArray[_sendNumber] forCharacteristic:self.TX_Characteristic type:CBCharacteristicWriteWithResponse];
                   NSLog(@"发送的包 %@",self.packArray[_sendNumber]);
                   _sendNumber++;
                   [self setValue:[NSString stringWithFormat:@"%d",_sendNumber] forKey:@"sendNumber"];
               }
               //发送至最后一包的时候
               else if ([string3 isEqualToString:@"<01000804 00>"] && _sendNumber == self.allSection-1) {
                   
                   [self.peripherale writeValue:self.lastData forCharacteristic:self.TX_Characteristic type:CBCharacteristicWriteWithResponse];
                   NSLog(@"发送了最后一条指令");
                   //确保进度条显示到100%
                   _sendNumber++;
                   [self setValue:[NSString stringWithFormat:@"%d",_sendNumber] forKey:@"sendNumber"];
                   //把包数重新归零
                   _sendNumber = 0;
                   NSLog(@"%lu %ld",(unsigned long)_dataArray.count,self.allSection);
               }
               
               
           }
           /*第三种大情况
            1.发送完毕 lastData 之后
            */
           else if ([str isEqualToString:@"<ab100000 00000700>"]) {
               
               NSString * string33 = [NSString stringWithFormat:@"%@",_dataArray[arrayCount-1]];
               
               if ([string33 isEqualToString:@"<01008001 0000>"] && (_dataArray.count-9)/3 == self.allSection)
               {
                   NSLog(@"蓝牙数据传输成功 %@",_dataArray.lastObject);
                   [DFULocalNotification registerLocalNotification:@"蓝牙数据传输完成"];
               }
               else if([string33 isEqualToString:@"<01008001 0000>"] == NO && (_dataArray.count-9)/3 == self.allSection)
               {
                   NSLog(@"蓝牙数据传输错误 %@",_dataArray.lastObject);
                   [DFULocalNotification registerLocalNotification:@"蓝牙数据传输错误"];
               }
           }
       }
    }
}

重点就是在这个回调函数里面,至于其他的文件解读,加密,校验什么的我就不上代码了,主要还是给大家提供一种思路吧!

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

推荐阅读更多精彩内容