Атомарный типСоответствующая специализация
atomic_boolstd::atomic
atomic_charstd::atomic
atomic_scharstd::atomic
atomic_uharstd::atomic
atomic_intstd::atomic
atomic_uintstd::atomic
atomic_shortstd::atomic
atomic_ushortstd::atomic
atomic_longstd::atomic
atomic_ulongstd::atomic
atomic_llongstd::atomic
atomic_ullongstd::atomic
atomic_char16_tstd::atomic
atomic_char32_tstd::atomic
atomic_wchar_tstd::atomic

Помимо основных атомарных типов, в стандартной библиотеке С++ определены также псевдонимы typedef для атомарных типов, соответствующих различным неатомарным библиотечным typedef, например std::size_t. Они перечислены в табл. 5.2.

Таблица 5.2. Соответствие между стандартными атомарными и встроенными typedef

Атомарный typedefСоответствующий typedef из стандартной библиотеки
atomic_int_least8_tint_least8_t
atomic_uint_least8_tuint_least8_t
atomic_int_least16_tint_least16_t
atomic_uint_least16_tuint_least16_t
atomic_int_least32_tint_least32_t
atomic_uint_least32_tuint_least32_t
atomic_int_least64_tint_least64_t
atomic_uint_least64_tuint_least64_t
atomic_int_fast8_tint_fast8_t
atomic_uint_fast8_tuint_fast8_t
atomic_int_fast16_tint_fast16_t
atomic_uint_fast16_tuint_fast16_t
atomic_int_fast32_tint_fast32_t
atomic_uint_fast32_tuint_fast32_t
atomic_int_fast64_tint_fast64_t
atomic_uint_fast64_tuint_fast64_t
atomic_intptr_tintptr_t
atomic_uintptr_tuintptr_t
atomic_size_tsize_t
atomic_ptrdiff_tptrdiff_t
atomic_intmax_tintmax_t
atomic_uintmax_tuintmax_t

Да уж, типов немало! Но есть простая закономерность — атомарный тип, соответствующий стандартному typedef T, имеет такое же имя с префиксом atomic_: atomic_T. То же самое относится и к встроенным типам с тем исключением, что signed сокращается до s, unsigned — до u, a long long — до llong. Вообще говоря, проще написать std::atomic для нужного вам типа T, чем пользоваться альтернативными именами.

Стандартные атомарные типы не допускают копирования и присваивания в обычном смысле, то есть не имеют копирующих конструкторов и операторов присваивания. Однако им все же можно присваивать значения соответствующих встроенных типов, и они поддерживают неявные преобразования в соответствующие встроенные типы. Кроме того, в них определены функции-члены load(), store(), exchange(), compare_exchange_weak() и compare_exchange_strong(). Поддерживаются также составные операторы присваивания (там, где это имеет смысл) +=, -=, *=, |= и т.д., а для целочисленных типов и специализаций std::atomic<> для указателей — еще и операторы ++ и --. Этим операторам соответствуют также именованные функции-члены с идентичной функциональностью: fetch_add(), fetch_or() и т.д. Операторы присваивания возвращают сохраненное значение, а именованные функции-члены — значение, которое объект имел до начала операции. Это позволяет избежать потенциальных проблем, связанных с тем, что обычно операторы присваивания возвращают ссылку на объект в левой части. Чтобы получить из такой ссылки сохраненное значение, программа должна была бы выполнить еще одну операцию чтения, но тогда между присваиванием и чтением другой поток мог бы модифицировать значение, открывая дорогу гонке.

Но шаблон класса std::atomic<> — не просто набор специализаций. В нем есть основной шаблон, который можно использовать для создания атомарного варианта пользовательского типа. Поскольку это обобщенный шаблон класса, определены только операции load(), store() (а также присваивание значения пользовательского типа и преобразования в пользовательский тип), exchange(), compare_exchange_weak() и compare_exchange_strong().

Перейти на страницу:

Похожие книги