Objective C 到 Swift 代码转换,似乎无法在 Swift 中正确处理字节

Objective C to Swift code conversion, can't seem to treat bytes correctly in Swift

我正在开发一个简单的 Swift 蓝牙心率监测器 iOS 应用程序。我发现 this 很棒的教程,其中包含 objective C 代码。我已将其转换为 Swift,并且正在从我的心率监测器获取数据。我的问题是我似乎无法正确访问和转换 Swift.

中的字节数据

这是 Objective C 代码:

// Instance method to get the heart rate BPM information
- (void) getHeartBPMData:(CBCharacteristic *)characteristic error:(NSError *)error
{
    // Get the Heart Rate Monitor BPM
    NSData *data = [characteristic value];      // 1
    const uint8_t *reportData = [data bytes];
    uint16_t bpm = 0;

    if ((reportData[0] & 0x01) == 0) {          // 2
        // Retrieve the BPM value for the Heart Rate Monitor
        bpm = reportData[1];
    }
    else {
        bpm = CFSwapInt16LittleToHost(*(uint16_t *)(&reportData[1]));  // 3
    }
    // Display the heart rate value to the UI if no error occurred
    if( (characteristic.value)  || !error ) {   // 4
        self.heartRate = bpm;
        self.heartRateBPM.text = [NSString stringWithFormat:@"%i bpm", bpm];
        self.heartRateBPM.font = [UIFont fontWithName:@"Futura-CondensedMedium" size:28];
        [self doHeartBeat];
        self.pulseTimer = [NSTimer scheduledTimerWithTimeInterval:(60. / self.heartRate) target:self selector:@selector(doHeartBeat) userInfo:nil repeats:NO];
    }
    return;
}

这是 Swift 代码:

func peripheral(peripheral: CBPeripheral!,
    didUpdateValueForCharacteristic characteristic: CBCharacteristic!,
    error: NSError!) -> String
{

    // Get the Heart Rate Monitor BPM
    var data = characteristic.value
    var reportData = data.bytes
    var bpm : UInt16
    var rawByte : UInt8
    var outputString = ""
    rawByte = UInt8(reportData[0])
    bpm = 0

    if ((rawByte & 0x01) == 0) {          // 2
        // Retrieve the BPM value for the Heart Rate Monitor
        bpm = UInt16( reportData[4] )
    }
    else {
        bpm = CFSwapInt16LittleToHost(UInt16(reportData[1]))
    }

    outputString = String(bpm)
    return outputString
}
const uint8_t *reportData = [data bytes];

转换为

let reportData = UnsafePointer<UInt8>(data.bytes)

然后 reportData 的类型为 UnsafePointer<UInt8>,您可以像 (Objective-)C:

那样访问它
if (reportData[0] & 0x01) == 0 { ... }

接下来,

bpm = reportData[1];

在Swift中几乎相同。您必须明确地从 UInt8 转换为 UInt16 因为 – 与 (Objective-)C 不同 – Swift 不会在类型之间隐式转换:

bpm = UInt16(reportData[1]) 

放在一起:

func getHeartBPMData(characteristic: CBCharacteristic!) {
    let data = characteristic.value
    let reportData = UnsafePointer<UInt8>(data.bytes)
    var bpm : UInt16
    if (reportData[0] & 0x01) == 0 {
        bpm = UInt16(reportData[1])
    } else {
        bpm = UnsafePointer<UInt16>(reportData + 1)[0]
        bpm = CFSwapInt16LittleToHost(bpm)
    }

    // ...
}

请注意,您的大部分变量都可以声明为 常量let,而不是 var。而不是

bpm = CFSwapInt16LittleToHost(bpm)

您也可以使用可用的 littleEndian: 构造函数 对于所有整数类型:

bpm = UInt16(littleEndian: bpm)

Swift 3/4 更新:

func getHeartBPMData(characteristic: CBCharacteristic) {
    guard  let reportData = characteristic.value else {
        return 
    }

    let bpm : UInt16
    if (reportData[0] & 0x01) == 0 {
        bpm = UInt16(reportData[1])
    } else {
        bpm = UInt16(littleEndian: reportData.subdata(in: 1..<3).withUnsafeBytes { [=18=].pointee } )
    }

    // ...
}