3 字节切片和有符号整数类型之间的转换

问题描述 投票:0回答:2

我有一个由 3 个字节(以 LE 排序)组成的切片,表示一个有符号整数,我想将其转换为任何整数类型,最好是

int32
,然后再转换回自身。

b := []byte{0x01, 0x00, 0x80}

我尝试使用

big.Int
来做到这一点,但它的
SetBytes()
Bytes()
方法似乎只适用于无符号整数类型,不管
big.Int
实例也可以存储有符号整数类型。

这是我正在尝试做的 Python 等效项:

b = b"\x01\x00\x80"
i = int.from_bytes(b, "little", signed=True)
b_again = int.to_bytes("little", signed=True)

编辑:这个问题不是 Go []byte to Little/Big-Endian Signed Integer or Float? 的重复。我尝试过应用那里的答案,但结果出乎意料。请参阅我的第一条评论。

go integer deserialization byte twos-complement
2个回答
3
投票

我有一个由 3 个字节(以 LE 排序)组成的切片,表示一个有符号整数,我想将其转换为任何整数类型,最好是 int32,然后再转换回自身。


package main

import "fmt"

func ByteLE3ToInt24(b []byte) int32 {
    i := uint32(b[0])<<8 | uint32(b[1])<<16 | uint32(b[2])<<24
    return int32(i) >> 8
}

func Int24ToByteLE3(i int32) []byte {
    return []byte{byte(i), byte(i >> 8), byte(i >> 16)}
}

func main() {
    b := []byte{0x01, 0x00, 0x80} // Negative
    fmt.Println(b)
    fmt.Println(ByteLE3ToInt24(b))
    fmt.Println(Int24ToByteLE3(ByteLE3ToInt24(b)))
    fmt.Println()
    b = []byte{0x01, 0x00, 0x00} // Positive
    fmt.Println(b)
    fmt.Println(ByteLE3ToInt24(b))
    fmt.Println(Int24ToByteLE3(ByteLE3ToInt24(b)))
}

https://go.dev/play/p/tI8E2kSXopZ

[1 0 128]
-8388607
[1 0 128]

[1 0 0]
1
[1 0 0]

-1
投票

Little-Endian 意味着数字

0x01020304
被编码为字节序列
[]byte{4, 3, 2, 1}

您的示例数字有三个字节,所以我假设您的意思是

0x00010080
(左侧填充零字节)。

有一个类型

encoding/binary.ByteOrder
,有两个实例
LittleEndian
BigEndian
,它们提供了与字节相互转换的函数:

package main

import (
    "encoding/binary"
    "fmt"
)

func main() {
    var b [4]byte
    binary.LittleEndian.PutUint32(b[:], 0x00010080)
    i32 := int32(binary.LittleEndian.Uint32(b[:]))
    fmt.Println(b)   // Prints [128 0 1 0].
    fmt.Println(i32) // Prints 65664 (which is 0x00010080).
}
© www.soinside.com 2019 - 2024. All rights reserved.