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

时间:2018-08-26 15:15:13

标签: arrays swift macos types nsdata

我正在编写一个应用程序,需要将不同数据类型的数组转换为数据数组,然后将其通过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

1 个答案:

答案 0 :(得分:1)

您正在看到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(), {$0.append($1.bytes)})
print(Array(data)) //->[42, 195, 245, 72, 64]

如果要编写扩展名:

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

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