Не используйте union
!
C ++ не разрешает пробивать шрифты через union
s!
Чтение из поля объединения, которое не было последним записанным полем, равно неопределенное поведение !
Многие компиляторы поддерживают это как расширения, но язык не дает никаких гарантий.
См. Этот ответ для более подробной информации:
https://stackoverflow.com/a/11996970
Есть только два действительных ответа, которые гарантированно будут переносимыми.
Первый ответ, если у вас есть доступ к системе, которая поддерживает C ++ 20,
должен использовать std::endian
из заголовка <type_traits>
.
(На момент написания C ++ 20 еще не был выпущен, но если что-то не повлияет на включение std::endian
, это будет предпочтительным способом проверки порядка байтов во время компиляции из C ++ 20 лет.)
C ++ 20 и далее
constexpr bool is_little_endian = (std::endian::native == std::endian::little);
До C ++ 20 единственный верный ответ - сохранить целое число, а затем проверить его первый байт через тип punning.
В отличие от использования union
s, это явно разрешено системой типов C ++.
Также важно помнить, что для оптимальной мобильности следует использовать static_cast
,
потому что reinterpret_cast
определяется реализацией.
Если программа пытается получить доступ к сохраненному значению объекта через glvalue другого, чем один из следующих типов, поведение не определено:
...
char
или unsigned char
тип.
C ++ 11 и далее
enum class endianness
{
little = 0,
big = 1,
};
inline endianness get_system_endianness()
{
const int value { 0x01 };
const void * address = static_cast<const void *>(&value);
const unsigned char * least_significant_address = static_cast<const unsigned char *>(address);
return (*least_significant_address == 0x01) ? endianness::little : endianness::big;
}
C ++ 11 и более (без перечисления)
inline bool is_system_little_endian()
{
const int value { 0x01 };
const void * address = static_cast<const void *>(&value);
const unsigned char * least_significant_address = static_cast<const unsigned char *>(address);
return (*least_significant_address == 0x01);
}
C ++ 98 / C ++ 03
inline bool is_system_little_endian()
{
const int value = 0x01;
const void * address = static_cast<const void *>(&value);
const unsigned char * least_significant_address = static_cast<const unsigned char *>(address);
return (*least_significant_address == 0x01);
}