使用签名数据按位解包

时间:2011-10-13 13:45:12

标签: c++ c bit-manipulation bitwise-operators

我一直在尝试一段时间的包装&将一些字符解包为整数。虽然有一些与此问题相关的主题,但我的问题与签名班次有关。我没有得到解开签名值的“技巧”,即:

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 值?

提前致谢。

3 个答案:

答案 0 :(得分:4)

在将其移出该类型的范围之后,您将c1投射到unsigned char ,因此投射的结果为零。你应该在转移之前进行演员表:

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

答案 1 :(得分:2)

(unsigned char)(c1 << 8)

这将

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

您不需要任何此类内容,因此您应该使用((unsigned char)c1 << 8)

答案 2 :(得分:0)

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

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

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