轉載請註明出處ios
http://blog.csdn.net/pony_maggie/article/details/26740237
編程
做者:小馬xcode
IOS學習也一段時間了,該上點乾貨了。前段時間研究了一下IOS藍牙通信相關的東西,把研究的一個成果給你們分享一下。app
一 項目背景框架
簡單介紹一下作的東西,設備是一個金融刷卡器,經過藍牙與iphone手機通信。手機端的app經過發送不一樣的指令(經過藍牙)控制刷卡器執行一些動做,好比讀磁條卡,讀金融ic卡等。上幾張圖容易理解一些:iphone
看了上面幾張圖,你應該大概瞭解這是個什麼東東了。函數
二 IOS 藍牙介紹oop
藍牙協議自己經歷了從1.0到4.0的升級演變, 最新的4.0以其低功耗著稱,因此通常也叫BLE(Bluetoothlow energy)。學習
iOS 有兩個框架支持藍牙與外設鏈接。一個是 ExternalAccessory。從ios3.0就開始支持,也是在iphone4s出來以前用的比較多的一種模式,可是它有個很差的地方,External Accessory須要拿到蘋果公司的MFI認證。測試
另外一個框架則是本文要介紹的CoreBluetooth,在iphone4s開始支持,專門用於與BLE設備通信(由於它的API都是基於BLE的)。這個不須要MFI,而且如今不少藍牙設備都支持4.0,因此也是在IOS比較推薦的一種開發方法。
三 CoreBluetooth介紹
CoreBluetooth框架的核心實際上是兩個東西,peripheral和central, 能夠理解成外設和中心。對應他們分別有一組相關的API和類,以下圖所示:
若是你要編程的設備是central那麼你大部分用到,反之亦然。在咱們這個示例中,金融刷卡器是peripheral,咱們的iphone手機 是central,因此我將大部分使用上圖中左邊部分的類。使用peripheral編程的例子也有不少,好比像用一個ipad和一個iphone通 訊,ipad能夠認爲是central,iphone端是peripheral,這種狀況下在iphone端就要使用上圖右邊部分的類來開發了。
四 服務和特徵
有個概念有必要先說明一下。什麼是服務和特徵呢(service and characteristic)?
每一個藍牙4.0的設備都是經過服務和特徵來展現本身的,一個設備必然包含一個或多個服務,每一個服務下面又包含若干個特徵。特徵是與外界交互的最小單位。好比說,一臺藍牙4.0設備,用特徵A來描述本身的出廠信息,用特徵B來與收發數據等。
服務和特徵都是用UUID來惟一標識的,UUID的概念若是不清楚請自行google,國際藍牙組織爲一些很典型的設備(好比測量心跳和血壓的設備)規定了標準的service UUID(特徵的UUID比較多,這裏就不列舉了),以下:
固然還有不少設備並不在這個標準列表裏,好比我用的這個金融刷卡器。藍牙設備硬件廠商一般都會提供他們的設備裏面各個服務(service)和特徵(characteristics)的功能,好比哪些是用來交互(讀寫),哪些可獲取模塊信息(只讀)等。
五 實現細節
做爲一箇中心要實現完整的通信,通常要通過這樣幾個步驟:
創建中心角色—掃描外設(discover)—鏈接外設(connect)—掃描外設中的服務和特徵(discover)—與外設作數據交互(explore and interact)—斷開鏈接(disconnect)。
1創建中心角色
首先在我本身類的頭文件中要包含CoreBluetooth的頭文件,並繼承兩個協議<CBCentralManagerDelegate,CBPeripheralDelegate>,代碼以下:
2掃描外設(discover)
代碼以下:
這個參數應該也是能夠指定特定的peripheral的UUID,那麼理論上這個central只會discover這個特定的設備,可是我實際測試發現,若是用特定的UUID傳參根本找不到任何設備,我用的代碼以下:
目前不清楚緣由,懷疑和設備自己在的廣播包有關。
3鏈接外設(connect)
當掃描到4.0的設備後,系統會經過回調函數告訴咱們設備的信息,而後咱們就能夠鏈接相應的設備,代碼以下:
4掃描外設中的服務和特徵(discover)
一樣的,當鏈接成功後,系統會經過回調函數告訴咱們,而後咱們就在這個回調裏去掃描設備下全部的服務和特徵,代碼以下:
一個設備裏的服務和特徵每每比較多,大部分狀況下咱們只是關心其中幾個,因此通常會在發現服務和特徵的回調裏去匹配咱們關心那些,好比下面的代碼:
相信你應該已經注意到了,回調函數都是以"did"開頭的,這些函數不用你調用,達到條件後系統後自動調用。
5與外設作數據交互(explore and interact)
發送數據很簡單,咱們能夠封裝一個以下的函數:
_testPeripheral和_writeCharacteristic是前面咱們保存的設備對象和能夠讀寫的特徵。
而後咱們能夠在外部調用它,好比固然我要觸發刷卡時,先組好數據包,而後調用發送函數:
數據的讀分爲兩種,一種是直接讀(reading directly),另一種是訂閱(subscribe)。從名字也能基本理解二者的不一樣。實際使用中具體用一種要看具體的應用場景以及特徵自己的屬性。前一個好理解,特徵自己的屬性是指什麼呢?特徵有個properties字段(characteristic.properties),它是一個整型值,有以下幾個定義:
好比說,你要交互的特徵,它的properties的值是0x10,表示你只能用訂閱的方式來接收數據。我這裏是用訂閱的方式,啓動訂閱的代碼以下:
當設備有數據返回時,一樣是經過一個系統回調通知我,以下所示:
6 斷開鏈接(disconnect)
這個比較簡單,只須要一個API就好了,代碼以下:
六 成果展現
上幾張效果圖,UI沒怎麼修飾,主要關注功能,實現了讀取磁道信息,與金融ic卡進行APDU交互等功能。
/////////**********************************************************/////////
#import "ViewController.h"
#import <CoreBluetooth/CoreBluetooth.h>
#define kServiceUUID @"3AF70026-AB81-4EF2-B90B-9C482B4812F1"
#define kCharacteristicUUID @"2F5A22C7-7AB5-446A-9F94-4E9B924BE508"
@interface ViewController ()<CBCentralManagerDelegate, CBPeripheralDelegate>{
CBCentralManager* _manager;
NSMutableData* _data;
CBPeripheral* _peripheral;
}
@end
@implementation ViewController
- (void)viewDidLoad {
//建立一箇中央
_manager = [[CBCentralManager alloc] initWithDelegate:self queue:nil];
}
- (void)centralManagerDidUpdateState:(CBCentralManager *)central{
if (central.state != CBCentralManagerStatePoweredOn) {
NSLog(@"藍牙未打開");
return;
}
//開始尋找全部的服務
[_manager scanForPeripheralsWithServices:@[[CBUUID UUIDWithString:kServiceUUID]] options:@{
CBCentralManagerScanOptionAllowDuplicatesKey:@YES
}];
}
//尋找到服務
- (void)centralManager:(CBCentralManager *)central didDiscoverPeripheral:(CBPeripheral *)peripheral advertisementData:(NSDictionary *)advertisementData RSSI:(NSNumber *)RSSI{
//中止尋找
[_manager stopScan];
if (_peripheral != peripheral) {
_peripheral = peripheral;
//開始鏈接周邊
[_manager connectPeripheral:_peripheral options:nil];
}
}
//鏈接周邊成功
- (void)centralManager:(CBCentralManager *)central didConnectPeripheral:(CBPeripheral *)peripheral{
[_data setLength:0];
_peripheral.delegate = self;
//鏈接周邊服務
[_peripheral discoverServices:@[[CBUUID UUIDWithString:kServiceUUID]]];
}
//鏈接周邊失敗
- (void)centralManager:(CBCentralManager *)central didFailToConnectPeripheral:(CBPeripheral *)peripheral error:(NSError *)error{
NSLog(@"鏈接失敗");
}
//鏈接周邊服務
- (void)peripheral:(CBPeripheral *)peripheral didDiscoverServices:(NSError *)error{
if (error) {
NSLog(@"錯誤的服務");
return;
}
//遍歷服務
for (CBService* service in peripheral.services) {
if ([service.UUID isEqual:[CBUUID UUIDWithString:kServiceUUID]]) {
//鏈接特徵
[_peripheral discoverCharacteristics:@[[CBUUID UUIDWithString:kCharacteristicUUID]] forService:service];
}
}
}
//發現特徵
- (void)peripheral:(CBPeripheral *)peripheral didDiscoverCharacteristicsForService:(CBService *)service error:(NSError *)error{
if (error) {
NSLog(@"鏈接特徵失敗");
return;
}
//遍歷特徵
if ([service.UUID isEqual:[CBUUID UUIDWithString:kServiceUUID]]) {
for (CBCharacteristic* characteristic in service.characteristics) {
if ([characteristic.UUID isEqual:[CBUUID UUIDWithString:kCharacteristicUUID]]) {
//開始監聽特徵
[_peripheral setNotifyValue:YES forCharacteristic:characteristic];
}
}
}
}
//監聽到特徵值更新
- (void)peripheral:(CBPeripheral *)peripheral didUpdateNotificationStateForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error{
if (error) {
NSLog(@"特徵值出錯");
return;
}
if (![characteristic.UUID isEqual:[CBUUID UUIDWithString:kCharacteristicUUID]]) {
return;
}
//若是有新值,讀取新的值
if (characteristic.isNotifying) {
[peripheral readValueForCharacteristic:characteristic];
}
}
//收到新值
- (void)peripheral:(CBPeripheral *)peripheral didUpdateValueForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error{
NSString* str = [[NSString alloc] initWithData:characteristic.value encoding:NSUTF8StringEncoding];
NSLog(@"%@", str);
}
@end
//////////////***********************//////////////////
#import "ViewController.h"
#import <CoreBluetooth/CoreBluetooth.h>
#define kServiceUUID @"3AF70026-AB81-4EF2-B90B-9C482B4812F1"
#define kCharacteristicUUID @"2F5A22C7-7AB5-446A-9F94-4E9B924BE508"
@interface ViewController ()<CBPeripheralManagerDelegate>{
CBPeripheralManager* _manager;
CBMutableService* _service;
CBMutableCharacteristic* _characteristic;
}
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
_manager = [[CBPeripheralManager alloc] initWithDelegate:self queue:nil];
}
- (void)centralManager:(CBCentralManager *)central didDisconnectPeripheral:(CBPeripheral *)peripheral error:(NSError *)error{
NSLog(@"發現外設");
}
//檢測中央設備狀態
- (void)peripheralManagerDidUpdateState:(CBPeripheralManager *)peripheral{
if (peripheral.state != CBCentralManagerStatePoweredOn){
NSLog(@"藍牙關閉");
return;
}
//開始中心服務
[self startService];
}
//開始中心服務
- (void)startService{
//經過uuid建立一個特徵
CBUUID* characteristicUUID = [CBUUID UUIDWithString:kCharacteristicUUID];
//第一個參數uuid,第二個參數決定這個特徵怎麼去用,第三個是是否加密
_characteristic = [[CBMutableCharacteristic alloc] initWithType:characteristicUUID properties:CBCharacteristicPropertyNotify value:nil permissions:CBAttributePermissionsReadable];
//建立一個服務uuid
CBUUID* serviceUUID = [CBUUID UUIDWithString:kServiceUUID];
//經過uuid建立服務
_service = [[CBMutableService alloc] initWithType:serviceUUID primary:YES];
//給服務設置特徵
[_service setCharacteristics:@[_characteristic]];
//使用這個服務
[_manager addService:_service];
}
//添加服務後的回調
- (void)peripheralManager:(CBPeripheralManager *)peripheral didAddService:(CBService *)service error:(NSError *)error{
//若是沒有錯誤,就能夠開始廣播服務了
if (error == nil) {
[_manager startAdvertising:@{
CBAdvertisementDataLocalNameKey:@"PKServer",
CBAdvertisementDataServiceUUIDsKey:@[[CBUUID UUIDWithString:kServiceUUID]]
}];
//[_manager updateValue:[@"pk" dataUsingEncoding:NSUTF8StringEncoding] forCharacteristic:_characteristic onSubscribedCentrals:]
}
}
//有人鏈接成功後調用
- (void)peripheralManager:(CBPeripheralManager *)peripheral central:(CBCentral *)central didSubscribeToCharacteristic:(CBCharacteristic *)characteristic{
NSLog(@"鏈接成功");
[_manager updateValue:[@"pk" dataUsingEncoding:NSUTF8StringEncoding] forCharacteristic:_characteristic onSubscribedCentrals:@[central]];
}
//斷開調用
- (void)peripheralManager:(CBPeripheralManager *)peripheral central:(CBCentral *)central didUnsubscribeFromCharacteristic:(CBCharacteristic *)characteristic{
NSLog(@"斷開");
}
@end