如何将不同类型的值转换为数据字节

How to convert different type values to data byte

我正在编写一个应用程序,我需要将不同数据类型的数组转换为一个数据数组,我将通过 UDP 端口将其发送到另一个设备。

到目前为止,我有以下代码:

class ViewController: NSViewController {
  override func viewDidLoad() {
    super.viewDidLoad()

    let a: UInt8 = 42

    var da:[UInt8] = [a]
    let data = Data(buffer: UnsafeBufferPointer(start: &da, count: da.count))
    let x = data.toArray(type: UInt8.self)
    print(x)
  }
}

到目前为止,这有效。我得到以下输出:[42]
如果我将 let a: UInt8 = 42 更改为 let a: Float = 3.14var da:[Float] = [a],我将得到 [195, 245, 72, 64].

的结果

以上都是我所期望的。问题出在以下代码中:

class ViewController: NSViewController {
  override func viewDidLoad() {
    super.viewDidLoad()

    let b: UInt8 = 42
    let a: Float = 3.14

    var da:[Any] = [a, b]
    let data = Data(buffer: UnsafeBufferPointer(start: &da, count: da.count))
    let x = data.toArray(type: UInt8.self)
    print(x)
  }
}

在这里,我希望得到 [42, 195, 245, 72, 64] 除了我得到以下结果:[42, 0, 0, 0, 0, 0, 0, 0, 176, 10, 10, 0, 0, 96, 0, 0, 104, 41, 172, 143, 255, 127, 0, 0, 32, 161, 114, 0, 1, 0, 0, 0, 195, 245, 72, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 96, 228, 32, 1, 1, 0, 0, 0, 168, 153, 114, 0, 1, 0, 0, 0].

有人可以向我解释为什么会发生这种情况以及我应该改变什么以获得预期的结果吗?

编辑: 虽然我可以跳过填充,但我在开发应用程序时试图伪造符合以下规范的服务器。
https://forums.codemasters.com/discussion/136948/f1-2018-udp-specification#latest

您看到的是 Any 的内存表示。

print(MemoryLayout<Any>.size) //-> 32

Any 类型的值在一个大小为 32 的隐藏结构中表示,它可能包含一些原始数据的原始值或引用类型实例或巨大结构的地址,以及一些元数据有关所含值类型的数据。

这不是简单的填充。


你可以这样写:

(如果需要,您可能需要自己控制 填充。)

protocol ByteRepresentable {
    var bytes: Data {get}
}
extension ByteRepresentable {
    var bytes: Data {
        var copiedSelf = self
        return Data(bytes: &copiedSelf, count: MemoryLayout.size(ofValue: self))
    }
}
extension UInt8: ByteRepresentable {}
extension Float: ByteRepresentable {}
//...

let a: UInt8 = 42
let b: Float = 3.14

let da: [ByteRepresentable] = [a, b]
var data = da.reduce(into: Data(), {[=11=].append(.bytes)})
print(Array(data)) //->[42, 195, 245, 72, 64]

如果要写扩展:

extension Array where Element == ByteRepresentable {
    func toData() -> Data {
        return self.reduce(into: Data(), {[=12=].append(.bytes)})
    }
}

print(Array(da.toData())) //->[42, 195, 245, 72, 64]