使用 std::vector<无符号字符的内容初始化结构>

Initializing a struct with the contents of a std::vector<unsigned char>

本文关键字:gt 初始化 结构 无符号 std vector lt 使用 字符      更新时间:2023-10-16

这看起来应该是简单直接的,但谷歌却很少。

一个干净的,现代的(c++ 11)初始化一个简单的文件头结构体的方法是什么?

typedef struct FooHeader {
    uint8_t    FooCount;
    uint8_t    BarCount;
    uint32_t   BazOffsets[4];
} FooHeader;

中包含std::vector<unsigned char> ?创建子向量并将其数据转换为头结构类型是一个好主意吗?

为了避免遇到打包、对齐和端序问题,最好在字节级别读取数据(在几乎所有现代硬件上,您可以假设8位字节,但打包经常在编译器之间改变(甚至只是在不同的编译标志之间),大端序和小端序计算机仍然很常见)。

这意味着你最好的选择是:

FooHeader load_FooHeader(std::vector<unsigned char> const &dat) {
    static_assert(
        std::numeric_limits<unsigned char>::digits == 8,
        "Assumes 8-bit bytes");
    FooHeader retv;
    retv.FooCount = dat[0];
    retv.BarCount = dat[1];
    //Start at the fifth byte, to allow for padding.
    //If you want to use a packed format, use index = 2;
    std::size_t index{4};
    for (std::size_t i{0}, iend{4}; i < iend; ++i) {
        retv.BarOffsets[i] = 0;
        //Adjust ordering depending on desired endianness.
        //Currently uses little endian.
        for (std::size_t j{0}, jend{4}; j < jend; ++j) {
            retv.BarOffsets[i] |= dat[index + i*4 + (3-j)] << (j*8);
        }
    }
    return retv;
}