說明:這次實踐的主要目的是簡單掌握藍牙的一些基本信息救崔,實現(xiàn)數(shù)據(jù)的接發(fā),中心與外設(shè)互相發(fā)送數(shù)據(jù)暂论,大數(shù)據(jù)分塊發(fā)送。最后為項目鏈接拌禾。
中心1.png
外設(shè)2.png
藍牙基礎(chǔ)(常見縮寫)
MFI: 專們?yōu)樘O果設(shè)備制作的設(shè)備取胎,詳見:關(guān)于MFi認證你所必須要知道的事情;
BLE:藍牙4.0設(shè)備因為低耗電湃窍,所以也叫做BLE闻蛀;
MFI:開發(fā)使用ExternalAccessory 框架;
4.0 BLE:開發(fā)使用CoreBluetooth 框架您市;
Peripheral:外設(shè)觉痛,被連接的設(shè)備為Peripheral;
Central:中心,發(fā)起連接的是Central茵休;
Service:服務(wù)薪棒,每個外設(shè)會有很多服務(wù),每個服務(wù)中包含很多字段榕莺,這些字段的權(quán)限一般分為 讀read俐芯,寫write,通知notiy幾種;
Characteristic:一個服務(wù)下面也可以存在多個特征钉鸯,特征可以理解成具體實現(xiàn)功能的窗口泼各,一般特征都會有value,也就是特征值亏拉,特征是與外界交互的最小單位扣蜻;
Description:每個特征可以對應(yīng)一個或多個Description用戶描述characteristic的信息或?qū)傩裕?UUID:可以理解成藍牙上的唯一標識符,為了區(qū)分不同的服務(wù)和特征及塘,或者給服務(wù)和特征取名字莽使,我們就用UUID來代表服務(wù)和特征;
快速生成UUID:
打開Mac OS X的Terminal.app笙僚,用uuidgen命令生成一個128bit的UUID
//服務(wù)id
#define TRANSFER_SERVICE_UUID @"0FB51F75-C9D5-45DC-BA61-065BD4A5E3E8"
//特征id
#define TRANSFER_CHARACTERISTIC_UP_UUID @"B678C8E2-9B1A-4952-A320-EF6D42F0831A"
-------------------------------
藍牙中心模式流程:
1. 建立中心角色
2. 掃描外設(shè)(discover)
3. 連接外設(shè)(connect)
4. 掃描外設(shè)中的服務(wù)和特征(discover)
5. 與外設(shè)做數(shù)據(jù)交互(explore and interact)
6. 訂閱Characteristic的通知
7. 斷開連接(disconnect)
中心:首先要導(dǎo)入藍牙庫
#import <CoreBluetooth/CoreBluetooth.h>
遵守協(xié)議
<CBCentralManagerDelegate,CBPeripheralDelegate>
1. 建立中心角色 創(chuàng)建CBCentralManager對象
_centralManager = [[CBCentralManager alloc] initWithDelegate:self queue:nil];
2. 掃描外設(shè)(discover)
//查看設(shè)備是否支持 芳肌,若支持則掃描
-(void)centralManagerDidUpdateState:(CBCentralManager *)central{
NSArray *uuidArray = [NSArray arrayWithObjects:[CBUUID UUIDWithString:TRANSFER_SERVICE_UUID],nil];
NSDictionary *optionsDic = [[NSDictionary alloc] initWithObjectsAndKeys:[NSNumber numberWithBool:NO],CBCentralManagerScanOptionAllowDuplicatesKey, nil];
[_centralManager scanForPeripheralsWithServices:uuidArray options:optionsDic];
}
3. 連接外設(shè)(connect)
-(void)centralManager:(CBCentralManager *)central didDiscoverPeripheral:(CBPeripheral *)peripheral advertisementData:(NSDictionary *)advertisementData RSSI:(NSNumber *)RSSI {
if (_discoveredPeripheral != peripheral) {
_discoveredPeripheral = peripheral;
[_centralManager connectPeripheral:peripheral options:nil];
}
}
4. 掃描外設(shè)中的服務(wù)和特征(discover)
//當連接成功后,系統(tǒng)會通過回調(diào)函數(shù)告訴我們肋层,然后我們就在這個回調(diào)里去掃描設(shè)備下所有的服務(wù)和特征
-(void)centralManager:(CBCentralManager *)central didConnectPeripheral:(CBPeripheral *)peripheral {
peripheral.delegate = self;
//指定服務(wù)
[peripheral discoverServices:[[NSArray alloc] initWithObjects:[CBUUID UUIDWithString:TRANSFER_SERVICE_UUID], nil]];
//[peripheral discoverServices:nil];
}
-4.1 獲取外設(shè)的services 6. 訂閱Characteristic的通知
-(void)peripheral:(CBPeripheral *)peripheral didDiscoverServices:(NSError *)error{
//處理我們需要的特征
for (int i=0; i<peripheral.services.count; i++) {
CBService *service = peripheral.services[i];
//指定需要查找的特征(對應(yīng)外設(shè)中訂閱特征)
[peripheral discoverCharacteristics:[[NSArray alloc] initWithObjects:[CBUUID UUIDWithString:TRANSFER_CHARACTERISTIC_Up_UUID],[CBUUID UUIDWithString:TRANSFER_CHARACTERISTIC_Image_UUID], nil] forService:service];
//[peripheral discoverCharacteristics:nil forService:service];
}
}
- 4.2 獲取外設(shè)的Characteristics,獲取Characteristics的值亿笤,獲取Characteristics的Descriptor和Descriptor的
// 5 已搜索到Characteristics
-(void)peripheral:(CBPeripheral *)peripheral didDiscoverCharacteristicsForService:(CBService *)service error:(NSError *)error {
//屬于那個服務(wù) 根據(jù)不同的特征執(zhí)行不同的命令
if ([service.UUID isEqual:[CBUUID UUIDWithString:TRANSFER_SERVICE_UUID]]) {
for (int i=0; i<service.characteristics.count; i++) {
CBCharacteristic *characteristic = service.characteristics[i];
if ([characteristic.UUID isEqual:[CBUUID UUIDWithString:TRANSFER_CHARACTERISTIC_Up_UUID]]) {
_upCharacteristic = characteristic;
//監(jiān)聽設(shè)備
[peripheral setNotifyValue:YES forCharacteristic:characteristic];
}
}
}
}
5. 與外設(shè)做數(shù)據(jù)交互(explore and interact)
///獲取外設(shè)發(fā)來的數(shù)據(jù),不論是read和notify,獲取數(shù)據(jù)都是從這個方法中讀取栋猖。
-(void)peripheral:(CBPeripheral *)peripheral didUpdateValueForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error {
NSData *dataValue = characteristic.value;
}
//給外設(shè)發(fā)數(shù)據(jù)
- (void)wrietPeripheral:(CBPeripheral *)peripheral characteristic:(CBCharacteristic *)characteristic value:(NSData *)data {
if(_imageCharacteristic.properties & CBCharacteristicPropertyWrite || _imageCharacteristic.properties & CBCharacteristicPropertyWriteWithoutResponse) {
[self.discoveredPeripheral writeValue:data forCharacteristic:_imageCharacteristic type:CBCharacteristicWriteWithResponse];
}
}
// 寫入數(shù)據(jù)后的回調(diào)方法
- (void)peripheral:(CBPeripheral *)peripheral didWriteValueForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error {
}
停止掃描
[_centralManager stopScan];
主動斷開設(shè)備
[_centralManager cancelPeripheralConnection:_discoveredPeripheral];
大數(shù)據(jù)净薛,分塊發(fā)包
- (void)actionSend:(UIButton *)btn {//發(fā)送大數(shù)據(jù) 分包處理
NSInteger BLE_SEND_MAX_LEN = 512;
imgName = [imgName isEqualToString:@"2.jpg"] ? @"3.jpg" : @"2.jpg";
testImgView.image = [UIImage imageNamed:imgName];
NSData *msgData = UIImageJPEGRepresentation([UIImage imageNamed:imgName], 1.0);
for (int i = 0; i < [msgData length]; i += BLE_SEND_MAX_LEN) {
// 預(yù)加 最大包長度,如果依然小于總數(shù)據(jù)長度蒲拉,可以取最大包數(shù)據(jù)大小
if ((i + BLE_SEND_MAX_LEN) < [msgData length]) {
NSString *rangeStr = [NSString stringWithFormat:@"%i,%li", i, (long)BLE_SEND_MAX_LEN];
NSData *subData = [msgData subdataWithRange:NSRangeFromString(rangeStr)];
[[HMBLECenterHandle sharedHMBLECenterHandle] wrietPeripheral:nil
characteristic:nil
value:subData];
//根據(jù)接收模塊的處理能力做相應(yīng)延時
usleep(20 * 1000);
}
else {
NSString *rangeStr = [NSString stringWithFormat:@"%i,%i", i, (int)([msgData length] - i)];
NSData *subData = [msgData subdataWithRange:NSRangeFromString(rangeStr)];
[[HMBLECenterHandle sharedHMBLECenterHandle] wrietPeripheral:nil
characteristic:nil
value:subData];
usleep(20 * 1000);
//發(fā)送結(jié)束標識
NSData *exoData = [@"exo" dataUsingEncoding:NSUTF8StringEncoding];
[[HMBLECenterHandle sharedHMBLECenterHandle] wrietPeripheral:nil
characteristic:nil
value:exoData];
}
}
}
外設(shè):首先要導(dǎo)入藍牙庫
#import <CoreBluetooth/CoreBluetooth.h>
遵守協(xié)議
<CBPeripheralManagerDelegate>
1. 建立外設(shè)角色 創(chuàng)建CBPeripheralManager對象
_peripheralManager = [[CBPeripheralManager alloc] initWithDelegate:self queue:nil];
2. 本地Peripheral設(shè)置服務(wù),特性,描述肃拜,權(quán)限等等
//1 查看設(shè)備是否支持 ,若支持則創(chuàng)建 服務(wù)和特征
-(void)peripheralManagerDidUpdateState:(CBPeripheralManager *)peripheral{
//一組特征值
_upCharacteristic = [[CBMutableCharacteristic alloc] initWithType:[CBUUID UUIDWithString:TRANSFER_CHARACTERISTIC_UP_UUID] properties:CBCharacteristicPropertyNotify | CBCharacteristicPropertyRead | CBCharacteristicPropertyWrite | CBCharacteristicPropertyWriteWithoutResponse value:nil permissions:CBAttributePermissionsReadable | CBAttributePermissionsWriteable];
//一個服務(wù)中添加多個特征值
transferService = [[CBMutableService alloc] initWithType:[CBUUID UUIDWithString:TRANSFER_SERVICE_UUID] primary:YES];
NSArray *characters = [[NSArray alloc] initWithObjects:_upCharacteristic, nil];
[transferService setCharacteristics:characters];
// 2 添加一個服務(wù)
[self.peripheralManager addService:transferService];
}
3. Peripheral發(fā)送廣告
// 3 會監(jiān)聽didAddService
-(void)peripheralManager:(CBPeripheralManager *)peripheral didAddService:(CBService *)service error:(NSError *)error{
NSLog(@"添加服務(wù)");
if (error != nil) {
NSLog(@"添加服務(wù)失敶仆拧: %@",error.localizedDescription);
} else {
// 開始廣播
[self.peripheralManager startAdvertising:@{ CBAdvertisementDataLocalNameKey : @"Service_name", CBAdvertisementDataServiceUUIDsKey :@[[CBUUID UUIDWithString:TRANSFER_SERVICE_UUID]] }];
}
}
4. 設(shè)置處理訂閱燃领、取消訂閱、
//當中央端連接上了此設(shè)備并訂閱了特征時會回調(diào) didSubscribeToCharacteristic
-(void)peripheralManager:(CBPeripheralManager *)peripheral central:(CBCentral *)central didSubscribeToCharacteristic:(CBCharacteristic *)characteristic {
NSLog(@"中心已經(jīng)預(yù)定了特征 --- %@",characteristic);
}
//當中央端取消訂閱時會調(diào)用didUnsubscribeFromCharacteristic
-(void)peripheralManager:(CBPeripheralManager *)peripheral central:(CBCentral *)central didUnsubscribeFromCharacteristic:(CBCharacteristic *)characteristic {
NSLog(@"中心沒有從特征預(yù)定 -- %@",characteristic);
}
4. 讀characteristic锦援、寫characteristic的委托方法
//當接收到中央端讀的請求時會調(diào)用didReceiveReadRequest
- (void)peripheralManager:(CBPeripheralManager *)peripheral didReceiveReadRequest:(CBATTRequest *)request {
if (request.characteristic.properties & CBCharacteristicPropertyRead) {
NSData *data = request.characteristic.value;
[request setValue:data];
[self.peripheralManager respondToRequest:request withResult:CBATTErrorSuccess];
} else {
[self.peripheralManager respondToRequest:request withResult:CBATTErrorReadNotPermitted];
}
}
//當接收到中央端寫的請求時會調(diào)用didReceiveWriteRequest
- (void)peripheralManager:(CBPeripheralManager *)peripheral didReceiveWriteRequests:(NSArray<CBATTRequest *> *)requests {
CBATTRequest *request = requests[0];
if (request.characteristic.properties & CBCharacteristicPropertyWrite) {
CBMutableCharacteristic *c = (CBMutableCharacteristic *)request.characteristic;
c.value = request.value;
NSData *data = c.value;
[self.peripheralManager respondToRequest:request withResult:CBATTErrorSuccess];
} else {
[self.peripheralManager respondToRequest:request withResult:CBATTErrorWriteNotPermitted];
}
}
4. 往中心發(fā)送數(shù)據(jù)
- (BOOL)updateValue:(NSData *)value characteristic:(CBMutableCharacteristic *)characteristic {
if (value && characteristic) {
return [self.peripheralManager updateValue:value forCharacteristic:characteristic onSubscribedCentrals:nil];
}
return NO;
}