空中升級又叫固件升級,指你手機從服務(wù)器下載下來的包或者數(shù)據(jù)艇纺,通過藍牙傳輸給你的外設(shè)升級固件怎静。如果你能把藍牙的基礎(chǔ)搞懂,其實也并不是很難黔衡,我在這里只不過提供一下思路消约。
空中升級略難的地方在于數(shù)據(jù)處理和交互,尤其要以怎樣簡單完整的代碼來實現(xiàn)數(shù)據(jù)的讀寫是重點员帮,這就需要你和硬件工程師的交流和你自己的邏輯思維了或粮。
在上代碼以前,說一下有關(guān)藍牙的傳輸速度的捞高,因為我開發(fā)中碰到較大數(shù)據(jù)的傳輸氯材,著實害我費了很多腦筋渣锦。
藍牙數(shù)據(jù)傳輸中有連接延遲。其是為了低功耗考慮氢哮,允許從機在跳頻過程中不理會主機的跳頻指令袋毙,繼續(xù)睡眠一段時間。而主機不能因為從機睡眠而認為其斷開連接了冗尤。其是1.25毫秒一個單位听盖。明顯,這個數(shù)值越小裂七,傳輸速度也高皆看。
藍牙BLE協(xié)議規(guī)定連接參數(shù)最小是5,即7.25毫秒背零;而Android手機規(guī)定連接參數(shù)最小是8腰吟,即10毫秒。iOS規(guī)定是16徙瓶,即20毫秒毛雇。
連接參數(shù)完全由主機決定,但從機可以發(fā)出更新參數(shù)申請侦镇,主機可以接受也可以拒絕灵疮。Android手機一部接受,而ios比較嚴格壳繁,拒絕的概率比較高震捣。
一般場景,連接參數(shù)設(shè)置16氮趋,即20毫秒伍派,一般的傳輸速率是50* 20 = 1000字節(jié)/每秒江耀。如果每個連接事件傳輸更多的包剩胁,可以獲得更高的傳輸速率。
但是以上這種方法并不能真正解決傳輸?shù)乃俣瓤炻楣敹嘁簿拖嗖?倍或者3倍昵观。最好的方法就是在與app每次給藍牙發(fā)送的包數(shù),通暢可能考慮到數(shù)據(jù)不丟失舌稀,都是一包一包的發(fā)送啊犬,但是在空中升級這里不得已包數(shù)必須要多一點,比如一次發(fā)送十包壁查,具體還是看你們硬件那邊怎么寫藍牙協(xié)議了觉至。
我下面的demo是這樣的一個過程:
1.發(fā)送給外設(shè)指令,我要空中升級
->2.外設(shè)給我回OK之后我發(fā)送一個隨機數(shù)(自定義了一種隨機算法)睡腿,驗證開始固件升級
->3.判斷隨機數(shù)無誤语御,準備發(fā)送打包好的數(shù)據(jù)
->4.真正發(fā)送打包好的數(shù)據(jù)(每次發(fā)送10包峻贮,一包20個字節(jié)),這里會重復(fù)N多次应闯,看你的原數(shù)據(jù)包有多大纤控;每次接到我發(fā)的包后,外設(shè)都會給我會OK否碉纺,我收到OK后才會發(fā)一下個數(shù)據(jù)包
->5.告訴外設(shè)我數(shù)據(jù)發(fā)送完畢船万,并發(fā)送一段指令(包括本次空中升級數(shù)據(jù)包的大小,還有加密參數(shù)什么的)
->6.外設(shè)給我回OK無誤后骨田,才算真正升級完成
//更新特征的value時調(diào)用
-(void)peripheral:(CBPeripheral *)peripheral didUpdateValueForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error
{
if (error) {
return;
}
//找到已經(jīng)訂閱的串口耿导,輸出看結(jié)果
if ([[characteristic.UUID UUIDString] isEqualToString:@"6E400003-B5A3-F393-E0A9-E50E24DCCA9E"]) {
NSLog(@"返回的結(jié)果是 = %@",characteristic.value);
[_dataArray addObject:characteristic.value];
NSInteger arrayCount = _dataArray.count;
//藍牙每次都會回三條數(shù)據(jù)
if (arrayCount%3 == 0) {
//返回的頭
NSString *str=[[NSString alloc]initWithFormat:@"%@",_dataArray[arrayCount-3]];
/*第一種大情況
1.發(fā)送固件升級指令
2.發(fā)送隨機數(shù)
3.驗證隨機數(shù)是否正確
*/
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;
//隨機數(shù)
if (data2.length == 7 ) {
randomData1 = [_dataArray[arrayCount-1] subdataWithRange:NSMakeRange(5, 1)];
randomData2 = [_dataArray[arrayCount-1] subdataWithRange:NSMakeRange(6, 1)];
}
//發(fā)起固件升級之后回的
if ([string3 isEqualToString:@"<01008204 00010000 00>"]) {
//寫入隨機數(shù)
[self.peripherale writeValue:self.randomData forCharacteristic:self.TX_Characteristic type:CBCharacteristicWriteWithResponse];
NSLog(@"寫入的隨機數(shù) %@",self.randomData);
}
//寫入隨機數(shù)之后回的
if ([randomData1 isEqualToData:[_calculateRandom subdataWithRange:NSMakeRange(13, 1)]] && [randomData2 isEqualToData:[_calculateRandom subdataWithRange:NSMakeRange(14, 1)]] && [keyHeadStr isEqualToString:@"<06>"]){
//隨機數(shù)驗證成功
[self.peripherale writeValue:_successData forCharacteristic:self.TX_Characteristic type:CBCharacteristicWriteWithResponse];
NSLog(@"隨機數(shù)驗證成功");
}
//隨機數(shù)驗證成功之后
if ([string3 isEqualToString:@"<01000501 0080>"]) {
//發(fā)送第一包數(shù)據(jù)包
[self.peripherale writeValue:self.packArray[_sendNumber] forCharacteristic:self.TX_Characteristic type:CBCharacteristicWriteWithResponse];
NSLog(@"發(fā)送的包 %@",self.packArray[_sendNumber]);
_sendNumber++;
[self setValue:[NSString stringWithFormat:@"%d",_sendNumber] forKey:@"sendNumber"];
}
}
/*第二種大情況
1.校驗發(fā)送的包是否收到了
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) {
//發(fā)送數(shù)據(jù)包
[self.peripherale writeValue:self.packArray[_sendNumber] forCharacteristic:self.TX_Characteristic type:CBCharacteristicWriteWithResponse];
NSLog(@"發(fā)送的包 %@",self.packArray[_sendNumber]);
_sendNumber++;
[self setValue:[NSString stringWithFormat:@"%d",_sendNumber] forKey:@"sendNumber"];
}
//發(fā)送至最后一包的時候
else if ([string3 isEqualToString:@"<01000804 00>"] && _sendNumber == self.allSection-1) {
[self.peripherale writeValue:self.lastData forCharacteristic:self.TX_Characteristic type:CBCharacteristicWriteWithResponse];
NSLog(@"發(fā)送了最后一條指令");
//確保進度條顯示到100%
_sendNumber++;
[self setValue:[NSString stringWithFormat:@"%d",_sendNumber] forKey:@"sendNumber"];
//把包數(shù)重新歸零
_sendNumber = 0;
NSLog(@"%lu %ld",(unsigned long)_dataArray.count,self.allSection);
}
}
/*第三種大情況
1.發(fā)送完畢 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(@"藍牙數(shù)據(jù)傳輸成功 %@",_dataArray.lastObject);
[DFULocalNotification registerLocalNotification:@"藍牙數(shù)據(jù)傳輸完成"];
}
else if([string33 isEqualToString:@"<01008001 0000>"] == NO && (_dataArray.count-9)/3 == self.allSection)
{
NSLog(@"藍牙數(shù)據(jù)傳輸錯誤 %@",_dataArray.lastObject);
[DFULocalNotification registerLocalNotification:@"藍牙數(shù)據(jù)傳輸錯誤"];
}
}
}
}
}
重點就是在這個回調(diào)函數(shù)里面,至于其他的文件解讀盛撑,加密碎节,校驗什么的我就不上代碼了,主要還是給大家提供一種思路吧抵卫!