iOS藍牙開發(4.0)詳解

来源:https://www.cnblogs.com/shuaibaobao/archive/2019/02/27/10444878.html
-Advertisement-
Play Games

最近由於項目需要, 一直在研究藍牙4.0,在這兒分享給大家, 望共同進步. 一、關於藍牙開發的一些重要的理論概念: 1.當前ios中開發藍牙所運用的系統庫是<CoreBluetooth/CoreBluetooth.h>。 2.藍牙外設必須為4.0及以上(2.0需要MFI認證),否則無法開發,藍牙4. ...


最近由於項目需要, 一直在研究藍牙4.0,在這兒分享給大家, 望共同進步.

一、關於藍牙開發的一些重要的理論概念:

1.當前ios中開發藍牙所運用的系統庫是<CoreBluetooth/CoreBluetooth.h>。

2.藍牙外設必須為4.0及以上(2.0需要MFI認證),否則無法開發,藍牙4.0設備因為低耗電,所以也叫做BLE。

3.CoreBluetooth框架的核心其實是兩個東西,peripheral和central, 可以理解成外設和中心,就是你的蘋果手機就是中心,外部藍牙稱為外設。

4.服務和特征(service and characteristic):簡而言之,外部藍牙中它有若幹個服務service(服務你可以理解為藍牙所擁有的能力),而每個服務service下擁有若幹個特征characteristic(特征你可以理解為解釋這個服務的屬性)。

5.Descriptor(描述)用來描述characteristic變數的屬性。例如,一個descriptor可以規定一個可讀的描述,或者一個characteristic變數可接受的範圍,或者一個characteristic變數特定的單位。

6.我們使用的藍牙模塊是在淘寶買的, 大概十多元一個, ios大概每次可以接受90個位元組, 安卓大概每次可以接收20個位元組, 具體數字可能會浮動, 應該是與藍牙模塊有關。

二、藍牙連接的主要步驟

     1、創建一個CBCentralManager實例來進行藍牙管理;

     2、搜索掃描外圍設備;

     3、連接外圍設備;

     4、獲得外圍設備的服務;

     5、獲得服務的特征;

     6、從外圍設備讀取數據;

     7、給外圍設備發送(寫入)數據。

 

三、代碼

// 加入許可權訪問, 否則上傳AppStore會因為許可權不足失敗

1. 初始化

#import <CoreBluetooth/CoreBluetooth.h>
self.centralManager = [[CBCentralManager alloc] initWithDelegate:self queue:nil];

2. 搜索掃描外圍設備

/**
 *  --  初始化成功自動調用
 *  --  必須實現的代理,用來返回創建的centralManager的狀態。
 *  --  註意:必須確認當前是CBCentralManagerStatePoweredOn狀態才可以調用掃描外設的方法:
 scanForPeripheralsWithServices
 */
- (void)centralManagerDidUpdateState:(CBCentralManager *)central{
    switch (central.state) {
        case CBCentralManagerStateUnknown:
            NSLog(@">>>CBCentralManagerStateUnknown");
            break;
        case CBCentralManagerStateResetting:
            NSLog(@">>>CBCentralManagerStateResetting");
            break;
        case CBCentralManagerStateUnsupported:
            NSLog(@">>>CBCentralManagerStateUnsupported");
            break;
        case CBCentralManagerStateUnauthorized:
            NSLog(@">>>CBCentralManagerStateUnauthorized");
            break;
        case CBCentralManagerStatePoweredOff:
            NSLog(@">>>CBCentralManagerStatePoweredOff");
            break;
        case CBCentralManagerStatePoweredOn:
        {
            NSLog(@">>>CBCentralManagerStatePoweredOn");
            // 開始掃描周圍的外設。
            /*
             -- 兩個參數為Nil表示預設掃描所有可見藍牙設備。
             -- 註意:第一個參數是用來掃描有指定服務的外設。然後有些外設的服務是相同的,比如都有FFF5服務,那麼都會發現;而有些外設的服務是不可見的,就會掃描不到設備。
             -- 成功掃描到外設後調用didDiscoverPeripheral
             */
            [self.centralManager scanForPeripheralsWithServices:nil options:nil];
        }
            break;
        default:
            break;
    }
}
#pragma mark 發現外設
- (void)centralManager:(CBCentralManager *)central didDiscoverPeripheral:(CBPeripheral *)peripheral advertisementData:(NSDictionary<NSString *,id> *)advertisementData RSSI:(NSNumber *)RSSI{
    NSLog(@"Find device:%@", [peripheral name]);
    if (![_deviceDic objectForKey:[peripheral name]]) {
        NSLog(@"Find device:%@", [peripheral name]);
        if (peripheral!=nil) {
            if ([peripheral name]!=nil) {
                if ([[peripheral name] hasPrefix:@"根據設備名過濾"]) {
                    [_deviceDic setObject:peripheral forKey:[peripheral name]];
                     // 停止掃描, 看需求決定要不要加
//                    [_centralManager stopScan];
                    // 將設備信息傳到外面的頁面(VC), 構成掃描到的設備列表
                    if ([self.delegate respondsToSelector:@selector(dataWithBluetoothDic:)]) {
                        [self.delegate dataWithBluetoothDic:_deviceDic];
                    }
                }
            }
        }
    }
}

3.連接外圍設備

// 連接設備(.h中聲明出去的介面, 一般在點擊設備列表連接時調用)
- (void)connectDeviceWithPeripheral:(CBPeripheral *)peripheral
{
    [self.centralManager connectPeripheral:peripheral options:nil];
}
#pragma mark 連接外設--成功
- (void)centralManager:(CBCentralManager *)central didConnectPeripheral:(CBPeripheral *)peripheral{
    //連接成功後停止掃描,節省記憶體
    [central stopScan];
    peripheral.delegate = self;
    self.peripheral = peripheral;
    //4.掃描外設的服務
    /**
     --     外設的服務、特征、描述等方法是CBPeripheralDelegate的內容,所以要先設置代理peripheral.delegate = self
     --     參數表示你關心的服務的UUID,比如我關心的是"FFE0",參數就可以為@[[CBUUID UUIDWithString:@"FFE0"]].那麼didDiscoverServices方法回調內容就只有這兩個UUID的服務,不會有其他多餘的內容,提高效率。nil表示掃描所有服務
     --     成功發現服務,回調didDiscoverServices
     */
    [peripheral discoverServices:@[[CBUUID UUIDWithString:@"你要用的服務UUID"]]];
    if ([self.delegate respondsToSelector:@selector(didConnectBle)]) {
       // 已經連接
        [self.delegate didConnectBle];
    }
}
#pragma mark 連接外設——失敗
- (void)centralManager:(CBCentralManager *)central didFailToConnectPeripheral:(CBPeripheral *)peripheral error:(NSError *)error{
    NSLog(@"%@", error);
}
#pragma mark 取消與外設的連接回調
- (void)centralManager:(CBCentralManager *)central didDisconnectPeripheral:(CBPeripheral *)peripheral error:(NSError *)error{
    NSLog(@"%@", peripheral);
}

4. 獲得外圍設備的服務

#pragma mark 發現服務回調
- (void)peripheral:(CBPeripheral *)peripheral didDiscoverServices:(NSError *)error{
    
    //NSLog(@"didDiscoverServices,Error:%@",error);
    CBService * __nullable findService = nil;
    // 遍歷服務
    for (CBService *service in peripheral.services)
    {
        //NSLog(@"UUID:%@",service.UUID);
        if ([[service UUID] isEqual:[CBUUID UUIDWithString:@"你要用的服務UUID"]])
        {
            findService = service;
        }
    }
    NSLog(@"Find Service:%@",findService);
    if (findService)
        [peripheral discoverCharacteristics:NULL forService:findService];
}

5、獲得服務的特征

#pragma mark 發現特征回調
/**
 --  發現特征後,可以根據特征的properties進行:讀readValueForCharacteristic、寫writeValue、訂閱通知setNotifyValue、掃描特征的描述discoverDescriptorsForCharacteristic。
 **/
- (void)peripheral:(CBPeripheral *)peripheral didDiscoverCharacteristicsForService:(CBService *)service error:(NSError *)error{
    for (CBCharacteristic *characteristic in service.characteristics) {
        if ([characteristic.UUID isEqual:[CBUUID UUIDWithString:@"你要用的特征UUID"]]) {
            
            /**
             -- 讀取成功回調didUpdateValueForCharacteristic
             */
            self.characteristic = characteristic;
            // 接收一次(是讀一次信息還是數據經常變實時接收視情況而定, 再決定使用哪個)
//            [peripheral readValueForCharacteristic:characteristic];
            // 訂閱, 實時接收
            [peripheral setNotifyValue:YES forCharacteristic:characteristic];
            
            // 發送下行指令(發送一條)
            NSData *data = [@"硬體工程師給我的指令, 發送給藍牙該指令, 藍牙會給我返回一條數據" dataUsingEncoding:NSUTF8StringEncoding];
            // 將指令寫入藍牙
                [self.peripheral writeValue:data forCharacteristic:characteristic type:CBCharacteristicWriteWithResponse];
        }

        /**
         -- 當發現characteristic有descriptor,回調didDiscoverDescriptorsForCharacteristic
         */
        [peripheral discoverDescriptorsForCharacteristic:characteristic];
    }
}

6.從外圍設備讀取數據

#pragma mark - 獲取值
- (void)peripheral:(CBPeripheral *)peripheral didUpdateValueForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error{
    // characteristic.value就是藍牙給我們的值(我這裡是json格式字元串)
    NSData *jsonData = [characteristic.value dataUsingEncoding:NSUTF8StringEncoding];
        NSDictionary *dataDic = [NSJSONSerialization JSONObjectWithData:jsonData options:NSJSONReadingMutableContainers error:nil];
    // 將字典傳出去就可以使用了
}

#pragma mark - 中心讀取外設實時數據
- (void)peripheral:(CBPeripheral *)peripheral didUpdateNotificationStateForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error{
    if (characteristic.isNotifying) {
        [peripheral readValueForCharacteristic:characteristic];
    } else { 
        NSLog(@"Notification stopped on %@.  Disconnecting", characteristic);
        NSLog(@"%@", characteristic);
        [self.centralManager cancelPeripheralConnection:peripheral];
    }
}

7. 給外圍設備發送(寫入)數據

// 上文中發現特征之後, 發送下行指令的時候其實就是向藍牙中寫入數據
// 例:
// 發送檢查藍牙命令
- (void)writeCheckBleWithBle
{
    _style = 1;
    // 發送下行指令(發送一條)
    NSData *data = [@"硬體工程師提供給你的指令, 類似於5E16010203...這種很長一串" dataUsingEncoding:NSUTF8StringEncoding];
    [self.peripheral writeValue:data forCharacteristic:self.characteristic type:CBCharacteristicWriteWithResponse];
}
#pragma mark 數據寫入成功回調
- (void)peripheral:(CBPeripheral *)peripheral didWriteValueForCharacteristic:(CBCharacteristic *)characteristic error:(NSError *)error{
    NSLog(@"寫入成功");
    if ([self.delegate respondsToSelector:@selector(didWriteSucessWithStyle:)]) {
        [self.delegate didWriteSucessWithStyle:_style];
    }
}

8. 另外

// 掃描設備
- (void)scanDevice
{
    if (_centralManager == nil) {
    self.centralManager = [[CBCentralManager alloc] initWithDelegate:self queue:nil];
    [_deviceDic removeAllObjects];
     }
}
#pragma mark - 斷開連接
- (void)disConnectPeripheral{
    /**
     -- 斷開連接後回調didDisconnectPeripheral
     -- 註意斷開後如果要重新掃描這個外設,需要重新調用[self.centralManager scanForPeripheralsWithServices:nil options:nil];
     */
    [self.centralManager cancelPeripheralConnection:self.peripheral];
}
#pragma mark - 停止掃描外設
- (void)stopScanPeripheral{
    [self.centralManager stopScan];
}

由於硬體方面剛開始用藍牙2.0跟我對接, 導致程式一直搜索不到設備.希望小伙伴們註意一下這個問題. 如果有不清楚的地方歡迎留言探討.


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 資料庫索引,到底是什麼做的? 問題1. 資料庫為什麼要設計索引? 圖書館存了1000W本圖書,要從中找到《架構師之路》,一本本查,要查到什麼時候去? 於是,圖書管理員設計了一套規則: (1)一樓放歷史類,二樓放文學類,三樓放IT類… (2)IT類,又分軟體類,硬體類… (3)軟體類,又按照書名音序排 ...
  • 1. redis事務 使用方法:方法為先發送multi命令告訴redis,下麵所有的命令屬於同一個事務,先不要執行,而是把他們暫時存起來,redis返回OK,然後後面執行需要放在同一個事務里的命令,可以看到每個命令都會返回QUEUED表示這幾條命令已經進入等待執行的事務隊列中了,當需要在同一個事務中 ...
  • 可以使用EXTRACT() 函數。(oracle和mysql都有該函數)語法: EXTRACT(unit FROM date)date 參數是合法的日期表達式。unit 參數可以是下列的值:YEAR\MONTH\WEEK\DAY\HOUR\MINUTE\SECOND (這裡只列出部分常用值)上面依次 ...
  • 摘要: 下文將分享三種不同的數據去重方法數據去重:需根據某一欄位來界定,當此欄位出現大於一行記錄時,我們就界定為此行數據存在重覆。 數據去重方法1: 當表中最在最大流水號時候,我們可以通過關聯的方式為每條重覆的記錄獲取唯一值數據去重方法2:為表中記錄,按照指定欄位進行群組,並獲取最大流水號,然後再進 ...
  • MySql中處理字元串時間,會預設把第一個數字當成年份處理。 在C#伺服器中,使用Date.Now.ToString()生成的字元串時間,如果不指定字元串格式,C#會按照系統語言輸出不同的字元串格式,如: a. 美國: 06/01/2019 01:59:00 PM b.中國: 2019/06/01 ...
  • 大家都知道,評論和評分是決定app在appstore中排名的重要因素,但是大部分用戶下載安裝APP後卻不會去點評,所以添加提示用戶去點評的功能是很必要的。 目前,AppStore點贊評分有兩種方法,一種是跳出應用,跳轉到AppStore;進行評分.另一種是在應用內,內置AppStore進行評分. 序 ...
  • 【目錄】 (一)上傳圖片到伺服器一 Android代碼 (二)上傳圖片到伺服器二 Android 系統7.0以上調用相機相容問題 (三)上傳圖片到伺服器三 後臺伺服器代碼 一、相關知識 ①Android許可權申請 ②網路訪問框架OKHttp ③記憶體溢出問題:圖片壓縮 ④Android 系統7.0以上調 ...
  • 1). 在block內部使用外部指針且會造成迴圈引用情況下,需要用__week修飾外部指針: __weak typeof(self) weakSelf = self; 2). 在block內部如果調用了延時函數還使用弱指針會取不到該指針,因為已經被銷毀了,需要在block內部再將弱指針重新強引用一下 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...