1

我发现当源是 wchar_t 数组(打包结构的成员)时,wcslen() 在 gcc 上返回不正确的结果(并且在 msvc 上正确)。我知道在 linux sizeof(wchar_t) == 4 和 windows 上它的 2 但仍然无法理解打包如何影响 wcslen() 函数。如果我将 wchar_t/wcslen 更改为 char/strlen 它会按预期工作。

#include <cstdint>
#include <cwchar>
#include <cstring>

#pragma pack(push, 1)

struct A
{
    uint8_t c;
};

struct B
{
    A  a;
    wchar_t buf[9];
};

#pragma pack(pop)


int main()
{
    const wchar_t* s = L"05.00.06";
    B b{};
    memcpy(b.buf, s, wcslen(s) * sizeof(wchar_t));

    return wcslen(b.buf);
}

为什么用 gcc 编译的这段代码返回 7?它应该返回 8(就像 msvc 一样)。顺便说一句,复制的字节是正确的(b.buf[7] == '6')。

4

1 回答 1

4

此代码的行为是未定义且不可预测的。您向wcslen函数传递了一个无效指针,因为它不一定满足其类型的对齐要求。

可能在您的平台上,对齐要求wchar_t是 2。因此您传递给的指针wcslen无效。您看不到类似的行为,strlen因为在这种情况下对齐要求是 1,这意味着根本没有要求。

除非您知道您已遵守平台的对齐要求,否则请勿对打包结构进行操作。否则,结果是完全不可预测的。在许多平台上,您的代码会崩溃。

于 2019-10-22T18:20:40.793 回答