Как быстро преобразовать Int16 в два байта UInt8

У меня есть некоторые двоичные данные, которые кодируют двухбайтовое значение в виде целого числа со знаком.

bytes[1] = 255  // 0xFF
bytes[2] = 251  // 0xF1

Декодирование

Это довольно просто - я могу извлечь значение Int16 из этих байтов с помощью:

Int16(bytes[1]) << 8 | Int16(bytes[2])

Кодирование

Здесь я сталкиваюсь с проблемами. Большая часть моей спецификации данных требует UInt, и это легко, но у меня возникают проблемы с извлечением двух байтов, составляющих Int16

let nv : Int16 = -15
UInt8(nv >> 8) // fail
UInt8(nv)      // fail

Вопрос

Как мне извлечь два байта, которые составляют значение Int16

Ответ 1

Вы должны работать с целыми целыми знаками:

let bytes: [UInt8] = [255, 251]
let uInt16Value = UInt16(bytes[0]) << 8 | UInt16(bytes[1])
let uInt8Value0 = uInt16Value >> 8
let uInt8Value1 = UInt8(uInt16Value & 0x00ff)

Если вы хотите конвертировать UInt16 в битовый эквивалент Int16, вы можете сделать это с помощью определенного инициализатора:

let int16Value: Int16 = -15
let uInt16Value = UInt16(bitPattern: int16Value)

И наоборот:

let uInt16Value: UInt16 = 65000
let int16Value = Int16(bitPattern: uInt16Value)

В вашем случае:

let nv: Int16 = -15
let uNv = UInt16(bitPattern: nv)

UInt8(uNv >> 8)
UInt8(uNv & 0x00ff)

Ответ 2

Вы можете использовать init(truncatingBitPattern: Int16) инициализатор:

let nv: Int16 = -15
UInt8(truncatingBitPattern: nv >> 8) // -> 255
UInt8(truncatingBitPattern: nv) // -> 241

Ответ 3

Я бы просто сделал это:

let a = UInt8(nv >> 8 & 0x00ff)  // 255
let b = UInt8(nv & 0x00ff)       // 241

Ответ 4

extension Int16 {
    var twoBytes : [UInt8] {
        let unsignedSelf = UInt16(bitPattern: self)
        return [UInt8(truncatingIfNeeded: unsignedSelf >> 8),
                UInt8(truncatingIfNeeded: unsignedSelf)]
    }
}

var test : Int16 = -15
test.twoBytes // [255, 241]