在Swift中将两个字节的UInt8数组转换为UInt16

zap*_*aph 29 swift

使用Swift我想将字节从uint8_t数组转换为整数.

"C"示例:

char bytes[2] = {0x01, 0x02};
NSData *data = [NSData dataWithBytes:bytes length:2];
NSLog(@"data: %@", data); // data: <0102>

uint16_t value2 = *(uint16_t *)data.bytes;
NSLog(@"value2: %i", value2); // value2: 513
Run Code Online (Sandbox Code Playgroud)

Swift尝试:

let bytes:[UInt8] = [0x01, 0x02]
println("bytes: \(bytes)") // bytes: [1, 2]
let data = NSData(bytes: bytes, length: 2)
println("data: \(data)") // data: <0102>

let integer1 = *data.bytes // This fails
let integer2 = *data.bytes as UInt16 // This fails

let dataBytePointer = UnsafePointer<UInt16>(data.bytes)
let integer3 = dataBytePointer as UInt16 // This fails
let integer4 = *dataBytePointer as UInt16 // This fails
let integer5 = *dataBytePointer // This fails
Run Code Online (Sandbox Code Playgroud)

从Swift中的UInt8数组创建UInt16值的正确语法或代码是什么?

我对NSData版本感兴趣,正在寻找一种不使用临时数组的解决方案.

Mar*_*n R 41

如果你想通过NSData那么它会像这样工作:

let bytes:[UInt8] = [0x01, 0x02]
println("bytes: \(bytes)") // bytes: [1, 2]
let data = NSData(bytes: bytes, length: 2)
print("data: \(data)") // data: <0102>

var u16 : UInt16 = 0 ; data.getBytes(&u16)
// Or:
let u16 = UnsafePointer<UInt16>(data.bytes).memory

println("u16: \(u16)") // u16: 513
Run Code Online (Sandbox Code Playgroud)

或者:

let bytes:[UInt8] = [0x01, 0x02]
let u16 = UnsafePointer<UInt16>(bytes).memory
print("u16: \(u16)") // u16: 513
Run Code Online (Sandbox Code Playgroud)

两种变体都假设字节是主机字节顺序.

更新Swift 3(Xcode 8):

let bytes: [UInt8] = [0x01, 0x02]
let u16 = UnsafePointer(bytes).withMemoryRebound(to: UInt16.self, capacity: 1) {
    $0.pointee
}
print("u16: \(u16)") // u16: 513
Run Code Online (Sandbox Code Playgroud)


Leo*_*bus 15

Swift 5 或更高版本中,您可以使用以下方法将字节转换[UInt8]UInt16withUnsafeBytes { $0.load(as: UInt16.self) }

let bytes: [UInt8] = [1, 2]
Run Code Online (Sandbox Code Playgroud)

加载为 UInt16

let uint16 = bytes.withUnsafeBytes { $0.load(as: UInt16.self) }    // 513 
Run Code Online (Sandbox Code Playgroud)

为了摆脱冗长,我们可以创建一个通用方法扩展ContiguousBytes

extension ContiguousBytes {
    func object<T>() -> T {
        withUnsafeBytes { $0.load(as: T.self) }
    }
}
Run Code Online (Sandbox Code Playgroud)

用法:

let bytes: [UInt8] = [1, 2]
let uint16: UInt16 = bytes.object()    // 513
Run Code Online (Sandbox Code Playgroud)

并访问集合中任何地方的字节:

extension Data {
    func subdata<R: RangeExpression>(in range: R) -> Self where R.Bound == Index {
        subdata(in: range.relative(to: self) )
    }
    func object<T>(at offset: Int) -> T {
        subdata(in: offset...).object()
    }
}
Run Code Online (Sandbox Code Playgroud)
extension Sequence where Element == UInt8  {
    var data: Data { .init(self) }
}
Run Code Online (Sandbox Code Playgroud)
extension Collection where Element == UInt8, Index == Int {
    func object<T>(at offset: Int = 0) -> T {
        data.object(at: offset)
    }
}
Run Code Online (Sandbox Code Playgroud)

用法:

let bytes: [UInt8] = [255, 255, 1, 2]
let uintMax: UInt16 = bytes.object()      // 65535 at offset zero
let uint16: UInt16 = bytes.object(at: 2)  // 513   at offset two
Run Code Online (Sandbox Code Playgroud)


pqn*_*net 5

我不知道 swift 的语法,但是像这样的东西呢:

let a:UInt16 = UInt16(bytes[0]) * 256 + UInt16(bytes[1])
Run Code Online (Sandbox Code Playgroud)

  • 关闭,这确实有效:`let a:UInt16 = UInt16(bytes[0]) * 256 + UInt16(bytes[1])`。 (2认同)

pjs*_*pjs 5

怎么样

let bytes:[UInt8] = [0x01, 0x02]
let result = (UInt16(bytes[1]) << 8) + UInt16(bytes[0])
Run Code Online (Sandbox Code Playgroud)

使用循环,这很容易推广到更大的字节数组,并且它可以包装在一个函数中以便于阅读:

let bytes:[UInt8] = [0x01, 0x02, 0x03, 0x04]

func bytesToUInt(byteArray: [UInt8]) -> UInt {
  assert(byteArray.count <= 4)
  var result: UInt = 0
  for idx in 0..<(byteArray.count) {
    let shiftAmount = UInt((byteArray.count) - idx - 1) * 8
    result += UInt(byteArray[idx]) << shiftAmount
  }
  return result
}

println(bytesToUInt(bytes))    // result is 16909060
Run Code Online (Sandbox Code Playgroud)