使用签名数据进行逐位解包

Bitwise unpacking using signed data

本文关键字:数据      更新时间:2023-10-16

我已经尝试了一段时间的pack&将一些字符解压缩为一个整数。虽然有一些话题与这个问题有关,但我的问题与签名轮班有关。我没有破解签名值的"窍门",即:

char c1 = -119;
char c2 = 26;
// pack
int packed = (unsigned char)c1 | (c2 << 8);
// unpack
c1 = packed >> 0;
c2 = packed >> 8;
// printf(c1, c2) -> Unpacked data: -119 | 26

这与预期一样有效,但当我尝试打包更多数据时,即:

char c0 = -42;
char c1 = -119;
char c2 = 26;
// pack
int packed = (unsigned char)c0 | (unsigned char)(c1 << 8) | (c2 << 16);
// unpack
c0 = packed >> 0;
c1 = packed >> 8;
c2 = packed >> 16;
// printf -> Unpacked data: -42 | 0 | 26

c1值丢失。我想这与有关,符号位被移到了高阶位置

我如何才能取回c1值?

提前谢谢。

在将c1移出该类型的范围后,将其强制转换为unsigned char,因此强制转换的结果为零。你应该在换班前做演员:

int packed = (unsigned char)c0 | ((unsigned char)c1 << 8) | (c2 << 16);
(unsigned char)(c1 << 8)

这将

  • 移位错误的(符号扩展(值
  • 将结果修剪为8位(产生0(

您不想要这些,所以应该使用((unsigned char)c1 << 8)

一些int是16位。要使此代码可移植,请使用int32_t。实现这一点的正确方法(如果有点偏执的话(是:

int32_t packed = ((uint8_t)c0) | (((uint8_t)c1)<<8) | (((uint8_t)c2) << 16);

我也倾向于按相反的顺序列出这些字符,所以哪些字符成为最重要和最不重要的字节更自然。