![](/img/trans.png)
[英]How can I reliably transpose an array of chars to an array of uint64_t and back again
[英]How can one store a uint64_t in an array?
我正在尝试将 crc64 校验和的uint64_t
表示形式存储为数组。
校验和总是像uint64_t res = 0x72e3daa0aa188782
,所以我希望将其存储为一个数组char digest[8]
,其中digest[0]
为 72, digest[1]
为 e3... digest[7]
是82。
我尝试循环/除以分解数字,但如果它是一个较小的 integer,并且如果起点是 Base-10,因为起点是 Base-16,则 output 应该但所描述的内容更合适以上。
更新:我删除了无意义的代码,并希望我能接受所有三个答案,因为他们都按照我的要求做了。 位移是我希望得到的答案,所以这就是它被接受的原因。
您应该使用除以 256,而不是除以 10:
unsigned char digest[sizeof(uint64_t) / sizeof(char)];
for (int i = sizeof(digest) - 1; i >= 0; i--) {
digest[i] = res % 256; // get the last byte
res /= 256; // get the remainder
}
// for demo purposes
for (int i = 0; i < sizeof(digest); i++) {
printf("%x ", digest[i]);
}
// 72 e3 da a0 aa 18 87 82
移位和按位与也可以做你需要的。 例如
unsigned char digest[8];
int shift = 56;
for (int i = 0; i < 8; ++i)
{
digest[i] = (res >> shift) & 0xff;
shift -= 8;
}
如果可以更改res
的值,另一种方法是:
for (int i = 7; i >= 0; --i)
{
digest[i] = res & 0xff;
res >>= 8;
}
显然,如果你除以 10,你会得到十进制数字,而不是你所期望的。 在这种情况下,您只想获取uint64_t
的底层位,这可以通过简单的memcpy
来完成。 但是您期望字节以大端顺序排列,因此您还需要先转换为大端
uint64_t number = 0x72e3daa0aa188782ull;
#ifdef __unix__
uint64_t number_be = htonll(number);
#else
uint64_t number_be = htobe64(number);
#endif
char digest[8];
memcpy(&digest, &number_be , sizeof number);
在 C 中也可以使用联合来完成
union Digest
{
uint64_t res;
char bytes[8];
} digest;
digest.res = htonll(0x72e3daa0aa188782ull); // or htobe64
// now just use digest.bytes[]
这些只需要几条机器指令,因此比为了获得 8 个字节而循环 8 次要快得多
我认为您可以在这里使用union
。
union foo {
uint64_t u64;
uint8_t u8[8];
};
这使您不必进行任何转换。 您可以使用foo.u64
访问 64 位值或使用foo.u8[0]
到foo.u8[7]
访问 8 位值。
尝试这个:
#define INIT_LIST \
201234567890123456, \
12345678901234567890, \
98765432109876543, \
65432109887, \
12345234512345, \
217631276371261627, \
12354123512453124, \
2163521442531, \
2341232142132321, \
1233432112
#define STR_(...) #__VA_ARGS__
#define STR(x) STR_(x)
int main (void)
{
uint64_t numbers[10] = { INIT_LIST };
char array[] = STR(INIT_LIST);
puts(array);
}
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.