备选方案#1:使用.pointee
的属性UnsafePointer
如以下问答中所述
您可以在 Swift 2.2 中使用,例如
UnsafePointer<UInt16>(bytes).memory
将 2 字节UInt8
数组转换为UInt16
值(主机字节顺序)。
现在,在 Swift 3.0-dev.memory
中已经被.pointee
. 应用上述以及此更改,我们同样可以使用UnsafePointer<UInt32>(bytes).pointee
来访问UInt32
4 字节UInt8
数组的表示,但请注意该表示使用主机字节顺序进行转换。为了可能(如有必要)将其转换为整数的大端表示,我们可以使用.bigEndian
每个整数类型可用的属性,如下面的问答中所述:
这在 Swift 3.0-dev 中仍然有效,因此,我们可以getUInt32BE(...)
如下构造您的方法
extension Collection where Iterator.Element == UInt8 {
public func getUInt32BE(at: Index.Distance) -> UInt32? {
let from = startIndex.advanced(by: at, limit: endIndex)
let to = from.advanced(by: 4, limit: endIndex)
guard case let bytes = Array(self[from..<to])
where bytes.count == 4 else { return nil }
return UnsafePointer<UInt32>(bytes).pointee.bigEndian
}
}
备选方案#2:使用位移
或者,使用带有位移的上述修改版本(类似于您的问题的尝试)而不是使用UnsafePointer<..>
:
extension Collection where Iterator.Element == UInt8 {
public func getUInt32BE(at: Index.Distance) -> UInt32? {
let from = startIndex.advanced(by: at, limit: endIndex)
let to = from.advanced(by: 4, limit: endIndex)
guard case let bytesSlice = self[from..<to]
where from.distance(to: to) == 4 else { return nil }
return bytesSlice.reduce(0) { (tot, val) -> UInt32 in
tot << 8 + UInt32(val as! UInt8)
}
}
}
val
请注意,我们需要通过强制转换from Iterator.Element
to来帮助编译器(由于扩展的子句,UInt8
这保证成功; )。where
Iterator.Element == UInt8
示例用法
上述两种选择中的任何一种的示例用法:
/* example usage */
let bytes: [UInt8] = [
0, // 0b 0000 0000
255, // 0b 1111 1111
0, // 0b 0000 0000
104, // 0b 0110 1000
76] // 0b 0100 1100
/* byteArr[1..<5], big-endian:
0b 1111 1111 0000 0000 0110 1000 0100 1100 */
let u32verify = 0b11111111000000000110100001001100
print(u32verify) // 4278216780
if let u32val = bytes.getUInt32BE(1) {
print(u32val) // 4278216780, OK
}
有关Collection
协议的详细信息(上述错误的原因是未使用Index.Distance
此协议的关联类型来构造您的子数组),请参见例如