空中升級又叫固件升級,指你手機從服務器下載下來的包或者數據,通過藍牙傳輸給你的外設升級固件。如果你能把藍牙的基礎搞懂,其實也並不是很難,我在這里只不過提供一下思路。
空中升級略難的地方在於數據處理和交互,尤其要以怎樣簡單完整的代碼來實現數據的讀寫是重點,這就需要你和硬件工程師的交流和你自己的邏輯思維了。
在上代碼以前,說一下有關藍牙的傳輸速度的,因為我開發中碰到較大數據的傳輸,着實害我費了很多腦筋。
藍牙數據傳輸中有連接延遲。其是為了低功耗考慮,允許從機在跳頻過程中不理會主機的跳頻指令,繼續睡眠一段時間。而主機不能因為從機睡眠而認為其斷開連接了。其是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:@"藍牙數據傳輸錯誤"]; } } } } }
重點就是在這個回調函數里面,至於其他的文件解讀,加密,校驗什么的我就不上代碼了,主要還是給大家提供一種思路吧!