2

在 CI 有

uint32 value = 39434;
uint8 firstByte = (unsigned char)value;
uint8 secondByte = (unsigned char)(value >> 8);

有没有可能在 Swift 中实现同样的目标?

4

2 回答 2

5

它在 Swift 中非常相似:

let value : UInt32 = 39434
let firstByte = UInt8(truncatingBitPattern: value) // 10
let secondByte = UInt8(truncatingBitPattern: value >> 8) // 154

这里需要特殊的初始化init(truncatingBitPattern:) 程序,因为 Swift(与 C 相比)不会隐式截断整数:

let firstByte = UInt8(value)

如果 value不适合UInt8.

另请参阅将 UInt32 快速拆分为 [UInt8] 以 获取可能的解决方案,该解决方案为您提供一个包含输入值的四个字节的数组。

于 2016-03-20T20:14:51.437 回答
1

bitpattern-dedicated initalizer 的替代方法init(truncatingBitPattern:)是手动屏蔽除感兴趣字节之外的所有字节,并适当地移动该字节;然后使用“标准”UInt8值转换初始化器:

let val : UInt32 = 39434

let byte1 = UInt8(val & 0x000000FF)         // 10
let byte2 = UInt8((val & 0x0000FF00) >> 8)  // 154
let byte3 = UInt8((val & 0x00FF0000) >> 16) // 0
let byte4 = UInt8((val & 0xFF000000) >> 24) // 0

/* ... */
let val : UInt32 = UINT32_MAX

let byte1 = UInt8(val & 0x000000FF)         // 255
let byte2 = UInt8((val & 0x0000FF00) >> 8)  // 255
let byte3 = UInt8((val & 0x00FF0000) >> 16) // 255
let byte4 = UInt8((val & 0xFF000000) >> 24) // 255

或者,为了简化,例如作为扩展中的计算属性

extension UInt32 {
    var asByteArray: [UInt8] {
        return [0, 8, 16, 24]
            .map { UInt8(self >> $0 & 0x000000FF) }
    }
}

/* example usage */
val.asByteArray 
    /* [10, 154, 0, 0], 
       [255, 255, 255, 255], for the examples above */

然而,这自然不如使用专用初始化程序安全,因为它将屏蔽和正确移位的责任放在了开发人员而不是编译器上。如果不注意,从四个字节到一个字节的非截断转换将产生运行时整数溢出。

于 2016-03-20T20:25:53.810 回答