Swift4.2使用CoreBluetooth进行蓝牙设备连接2019-04-17

代码完整,只要用自己 设备的名字 跟 特征uuid 即可看到效果

import UIKit
import CoreBluetooth

class PeripheralInfo {
    var serviceUUID: CBUUID?
    var characteristics: [CBCharacteristic]?
}
class TTBluetoothManger:NSObject {

    static var share =  TTBluetoothManger()
    private override init(){}
    
    lazy var centralManager:CBCentralManager = {
        let c =  CBCentralManager.init()
        c.delegate = self
        return c
    }()
    
    var currentPeripheral:CBPeripheral?
    var currentCharacteristic_01_01:CBCharacteristic?
    var currentCharacteristic_02_01:CBCharacteristic?

    ///设备名
    let Cperipheral_name = "M*****  "
    
    ///服务UUID
    let Cserve_01_uuid = "38*"
    let Cserve_02_uuid = "26*"
    
    ///特征UUID
    let  Ccharacteristic_01_01_uuid = "3***"
    let  Ccharacteristic_02_01_uuid = "2***"

  注:每个设备的服务与特征数量都不同

}

extension TTBluetoothManger{

    ///启用蓝牙,搜索链接设备
    ///在控制器中调用即可进行整个流程
    func bluetoohStar() {
        self.centralManager.delegate = self
    }
    
    func printShow(str:String) {
        
        print("=====================================")
        print("|             \(str)                |")
        print("-------------------------------------")
    }
    
    ///App向设备写入数据时调用次方法
    func deviceStartWriteValue(_ characteristic: CBCharacteristic) {
        
        ///这是我自己设备的写入数据
        let data = Data.init(bytes:[
            0x01,0xfe,0x00,0x00,
            0x23,0x33,0x10,0x00,
            0x64,0x00,0x00,0x00,
            0x00,0x00,0x00,0x00])
        currentPeripheral!.writeValue(data, for: characteristic, type: CBCharacteristicWriteType.withResponse)
    }
    
}

extension TTBluetoothManger:CBCentralManagerDelegate{
    func centralManagerDidUpdateState(_ central: CBCentralManager) {
        
        switch central.state {
        case .unknown:
            print("CBCentralManager state:", "unknown")
        case .resetting:
            print("CBCentralManager state:", "resetting")
        case .unsupported:
            print("CBCentralManager state:", "unsupported")
        case .unauthorized:
            print("CBCentralManager state:", "unauthorized")
        case .poweredOn:
            print("CBCentralManager state:", "poweredOn")
            ///扫描设备
            central.scanForPeripherals(withServices: nil, options: [CBCentralManagerScanOptionAllowDuplicatesKey:NSNumber.init(value: false)])
            
        case .poweredOff:
            print("CBCentralManager state:", "poweredOff")
        default:
            print("未知错误")
        }
    }
    
    ///发现设备
    func centralManager(_ central: CBCentralManager, didDiscover peripheral: CBPeripheral, advertisementData: [String : Any], rssi RSSI: NSNumber) {
        print("---------------------START---------------------")
        print("peripheral.name = \(peripheral.name ?? "搜索失败!")")
        if peripheral.name != nil{
            guard peripheral.name == "DMK28  " else{return}
            print("peripheral.name = \(peripheral.name!)")
            print("central = \(central)")
            print("peripheral = \(peripheral)")
            print("RSSI = \(RSSI)")
            print("advertisementData = \(advertisementData)")
            self.currentPeripheral = peripheral
            
            ///连接设备
            if let _ = self.currentPeripheral{
                central.stopScan()
                central.connect(self.currentPeripheral!, options: nil)
            }
            
        }
    }
    
    ///连接设备成功
    func centralManager(_ central: CBCentralManager, didConnect peripheral: CBPeripheral) {
        
        printShow(str: "连接成功")
        self.currentPeripheral = peripheral
        peripheral.delegate = self
        //开始寻找Services。传入nil是寻找所有Services
        peripheral.discoverServices(nil)
    }
    
    ///连接设备失败
    func centralManager(_ central: CBCentralManager, didFailToConnect peripheral: CBPeripheral, error: Error?) {
    
        printShow(str: "连接失败:\(error.debugDescription)")
    }

    ///断开连接
    func centralManager(_ central: CBCentralManager, didDisconnectPeripheral peripheral: CBPeripheral, error: Error?) {
        printShow(str: "断开连接")
        
        ///可重新扫描
    }
    
    
}

extension TTBluetoothManger:CBPeripheralDelegate{
    
    ///寻找服务
    func peripheral(_ peripheral: CBPeripheral, didDiscoverServices error: Error?) {
        printShow(str: "搜索服务")
        if error != nil{  print("服务异常:",error.debugDescription);return}
        guard let pservices = peripheral.services else {return}
        for ser in pservices {
            print("[服务的UUID] \(ser.uuid)")
            //在感兴趣的服务中寻找感兴趣的特征
            if ser.uuid.uuidString == Cserve_01_uuid || ser.uuid.uuidString == Cserve_02_uuid{
                self.currentPeripheral?.discoverCharacteristics(nil, for: ser)
            }
            
        }
    }

    /// 从感兴趣的服务中,确认 我们所发现感兴趣的特征
    func peripheral(_ peripheral: CBPeripheral, didDiscoverCharacteristicsFor service: CBService, error: Error?) {
        printShow(str: "确认特征")

        if error != nil{  print("特征异常:",error.debugDescription);return}
        guard let serviceCharacters = service.characteristics else {return}
        for characteristic in serviceCharacters {
            let characteristic_uuid = characteristic.uuid
            print("<特征UUID>",characteristic_uuid)

            // 订阅关于感兴趣特征的持续通知;
            // “当你启用特征值的通知时,外围设备调用……
            if characteristic_uuid.uuidString == Ccharacteristic_01_01_uuid{
                self.currentCharacteristic_01_01 = characteristic
                peripheral.setNotifyValue(true, for: characteristic)
            }
            //读取感兴趣的特征
            if characteristic_uuid.uuidString == Ccharacteristic_02_01_uuid{
                self.currentCharacteristic_02_01 = characteristic
                peripheral.readValue(for: characteristic)

            }
        }
    }
    
    //MARK: - 检测向外设写数据是否成功
    func peripheral(_ peripheral: CBPeripheral, didWriteValueFor characteristic: CBCharacteristic, error: Error?) {
        if error != nil {
            printShow(str: "写数据失败!!!")
        }else{printShow(str: "🌺写入数据成功🌺")}
        
    }
    
    // 接收外设发来的数据 每当一个特征值定期更新或者发布一次时,我们都会收到通知;
    // 阅读并解译我们订阅的特征值
    // MARK: - 获取外设发来的数据
    // 注意,所有的,不管是 read , notify 的特征的值都是在这里读取
    func peripheral(_ peripheral: CBPeripheral, didUpdateValueFor characteristic: CBCharacteristic, error: Error?) {
        printShow(str: "接收数据")
        if (characteristic.value != nil) {
            let data = characteristic.value
            
            let mod = Model()
            mod.read_analyzeData(fromData: data!)
        }
    }
    
    //接收characteristic信息    //MARK: - 特征的订阅状体发生变化
    func peripheral(_ peripheral: CBPeripheral, didUpdateNotificationStateFor characteristic: CBCharacteristic, error: Error?) {
        print("========特征的订阅状体变化========")
        printShow(str: characteristic.uuid.uuidString)
        
    }
    
}


class Model: NSObject {
    
    var waterArr = [Double]()
    var oilArr = [Double]()
    var lastwaterArr = [Double]()
    var lastoilArr = [Double]()
    
    var min:UInt8 = 0
    var sec:UInt8 = 0
    var waterValue:UInt8 = 0
    var oilValue:UInt8 = 0
    var countdownFlag:UInt8 = 0
    var workStatus:UInt8 = 0
    var surplus:UInt8 = 0
    var timeString:String = "00:00"
    var isPush = false
    
    var waterValue_100: Int = 0
    var oilValue_100:Int = 0
    
    func read_analyzeData(fromData data:Data) -> Void {
        
        let byteArr = data.bytes
        print(byteArr);
        if byteArr.count > 0{
            let hour = byteArr[7];
            let min = byteArr[8];
            let sec = byteArr[9];
            
            let waterValue = byteArr[10];
            let oilValue = byteArr[11];
            
            let countdownFlag = byteArr[12];
            let workStatus = byteArr[13];
            ///剩余使用次数
            let surplus = byteArr[14];
            
            
            print("[","时 = ",hour);
            print("分 = ",min);
            print("秒 = ",sec);
            print("水分 = ",waterValue);
            print("油分 = ",oilValue);
            print("倒计时标志 = ",countdownFlag);
            print("工作状态 = ",workStatus);
            print("剩余次数 = ",surplus,"]");
            
            var minString = "\(min)"
            var secString = "\(sec)"
            
            if min < 10 {
                minString = "0" + minString;
            }
            
            if sec < 10 {
                secString = "0" + secString;
            }
            
            let timeString = minString + ":" + secString;
            
            self.min = min;
            self.sec = sec;
            self.waterValue = waterValue;
            self.oilValue = oilValue;
            self.countdownFlag = countdownFlag;
            self.workStatus = workStatus;
            self.surplus = surplus;
            self.timeString = timeString;
            
            
            let temp_water = Int(waterValue)
            self.waterValue_100 = temp_water
            
            let temp_oil = Int(oilValue)
            self.oilValue_100 = temp_oil;
        }
    }
}










extension Data {
    /// Data -> Array, Dictionary
    ///
    /// - Returns: Array
    func toArray() -> [Any]? {
        
        return toArrayOrDictionary() as? [Any]
    }
    
    /// Data -> Array, Dictionary
    ///
    /// - Returns: Array
    func toDictionary() -> [String:Any]? {
        
        return toArrayOrDictionary() as? [String:Any]
    }
    
    /// Data -> Array, Dictionary
    ///
    /// - Returns: Any
    fileprivate func toArrayOrDictionary() -> Any? {
        
        do {
            
            let data = try JSONSerialization.jsonObject(with: self, options: JSONSerialization.ReadingOptions.allowFragments)
            
            return data
        } catch let _ {
            return nil
        }
    }
    
    public var bytes: Array<UInt8> {
        return Array(self)
    }
}

extension String {
    
    //16进制的转换
    //16进制类型的字符串[A-F,0-9]和Data之间的转换可以使用下面的方法。如果是包含=之类的可以直接用字符串转换Data即可

    ///16进制字符串转Data
    func hexData() -> Data? {
        var data = Data(capacity: count / 2)
        let regex = try! NSRegularExpression(pattern: "[0-9a-f]{1,2}", options: .caseInsensitive)
        regex.enumerateMatches(in: self, range: NSMakeRange(0, utf16.count)) { match, flags, stop in
            let byteString = (self as NSString).substring(with: match!.range)
            var num = UInt8(byteString, radix: 16)!
            data.append(&num, count: 1)
        }
        guard data.count > 0 else { return nil }
        return data
    }
    
    func utf8Data()-> Data? {
        return self.data(using: .utf8)
    }
    
}

extension Data {
    ///Data转16进制字符串
    func hexString() -> String {
        return map { String(format: "%02x", $0) }.joined(separator: "").uppercased()
    }
}

···
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,951评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,606评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,601评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,478评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,565评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,587评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,590评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,337评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,785评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,096评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,273评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,935评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,578评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,199评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,440评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,163评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,133评论 2 352