std::not_fn
| Определено в заголовочном файле <functional>
|
||
template< class F > /* неуказано */ not_fn( F&& f ); |
(1) | (начиная с C++17) (constexpr начиная с C++20) |
template< auto ConstFn > constexpr /* неуказано */ not_fn() noexcept; |
(2) | (начиная с C++26) |
ConstFn является нулевым указателем или нулевым указателем на элемент.Параметры
| f | — | объект, из которого создаётся объект Callable, удерживаемый оболочкой |
| Требования к типам | ||
- должен соответствовать требованиям Callable и MoveConstructible.
| ||
-std::is_constructible_v<std::decay_t<F>, F> должен быть true
| ||
Возвращаемое значение
T. Он имеет следующие элементы.
std::not_fn тип возвращаемого значения
Объекты-элементы
Тип возвращаемого значения std::not_fn содержит объект-элемент типа std::decay_t<F>.
Конструкторы
<tbody> </tbody> explicit T( F&& f ); |
(1) | (начиная с C++17) (constexpr начиная с C++20) (только для пояснения*) |
T( T&& f ) = default; T( const T& f ) = default; |
(2) | (начиная с C++17) |
std::decay_t<F>) из std::forward<F>(f). Генерирует любое исключение, созданное выбранным конструкторомstd::decay_t<F> должен быть MoveConstructible, возвращаемая оболочка вызова всегда MoveConstructible и CopyConstructible, если std::decay_t<F> является CopyConstructible.
|
Определения, заданные явно по умолчанию, делают тип возвращаемого значения не присваиваемым. |
|
|
Не указано, заданы ли эти конструкторы явно по умолчанию и можно ли присваивать тип возвращаемого значения. |
Функция-элемент operator()
<tbody>
</tbody>
<tbody class="t-dcl-rev t-dcl-rev-num ">
</tbody><tbody>
</tbody>
<tbody class="t-dcl-rev t-dcl-rev-num ">
</tbody><tbody>
</tbody>| (1) | ||
template<class... Args> auto operator()(Args&&... args) & -> decltype( !std::declval<std::invoke_result_t<std::decay_t<F>&, Args...>>()); template<class... Args> auto operator()(Args&&... args) const& -> decltype( !std::declval<std::invoke_result_t<std::decay_t<F> const&, Args...>>()); |
(начиная с C++17) (до C++20) |
|
template<class... Args> constexpr auto operator()(Args&&... args) & noexcept(/*смотрите ниже*/) -> decltype( !std::declval<std::invoke_result_t<std::decay_t<F>&, Args...>>()); template<class... Args> constexpr auto operator()(Args&&... args) const& noexcept(/*смотрите ниже*/) -> decltype( !std::declval<std::invoke_result_t<std::decay_t<F> const&, Args...>>()); |
(начиная с C++20) | |
| (2) | ||
template<class... Args> auto operator()(Args&&... args) && -> decltype( !std::declval<std::invoke_result_t<std::decay_t<F>, Args...>>()); template<class... Args> auto operator()(Args&&... args) const&& -> decltype( !std::declval<std::invoke_result_t<std::decay_t<F> const, Args...>>()); |
(начиная с C++17) (до C++20) |
|
template<class... Args> constexpr auto operator()(Args&&... args) && noexcept(/*смотрите ниже*/) -> decltype( !std::declval<std::invoke_result_t<std::decay_t<F>, Args...>>()); template<class... Args> constexpr auto operator()(Args&&... args) const&& noexcept(/*смотрите ниже*/) -> decltype( !std::declval<std::invoke_result_t<std::decay_t<F> const, Args...>>()); |
(начиная с C++20) | |
Пусть fd будет объектом-элементом типа std::decay_t<F>.
|
1) Эквивалент
return !std::invoke(fd, std::forward<Args>(args)...);2) Эквивалент
return !std::invoke(std::move(fd), std::forward<Args>(args)...);Если при вызове результата замена в тип возвращаемого значения первоначально выбранной перегрузки |
(начиная с C++17) (до C++20) |
|
Если во время вызова результата замена в тип возвращаемого значения первоначально выбранной перегрузки |
(начиная с C++20) |
std::not_fn возвращаемый тип без сохранения состояния
Тип возвращаемого значения это CopyConstructible класс без сохранения состояния. Не указано, является ли тип возвращаемого значения присваиваемым.
Функция-элемент operator()
<tbody>
</tbody> template< class... Args > constexpr auto operator()( Args&&... args ) const noexcept(/* смотрите ниже */) -> decltype( !std::declval<std::invoke_result_t<decltype((ConstFn)), Args...>>()); |
(начиная с C++26) | |
Выражение эквивалентно !std::invoke(ConstFn, std::forward<Args>(args)...).
Исключения
Исключения не генерируются, если их не генерирует создание fd.
Возможная реализация
| (1) not_fn |
|---|
namespace detail
{
template<class V, class F, class... Args>
constexpr bool negate_invocable_impl = false;
template<class F, class... Args>
constexpr bool negate_invocable_impl<std::void_t<decltype(
!std::invoke(std::declval<F>(), std::declval<Args>()...))>, F, Args...> = true;
template<class F, class... Args>
constexpr bool negate_invocable_v = negate_invocable_impl<void, F, Args...>;
template<class F>
struct not_fn_t
{
F f;
template<class... Args,
std::enable_if_t<negate_invocable_v<F&, Args...>, int> = 0>
constexpr decltype(auto) operator()(Args&&... args) &
noexcept(noexcept(!std::invoke(f, std::forward<Args>(args)...)))
{
return !std::invoke(f, std::forward<Args>(args)...);
}
template<class... Args,
std::enable_if_t<negate_invocable_v<const F&, Args...>, int> = 0>
constexpr decltype(auto) operator()(Args&&... args) const&
noexcept(noexcept(!std::invoke(f, std::forward<Args>(args)...)))
{
return !std::invoke(f, std::forward<Args>(args)...);
}
template<class... Args,
std::enable_if_t<negate_invocable_v<F, Args...>, int> = 0>
constexpr decltype(auto) operator()(Args&&... args) &&
noexcept(noexcept(!std::invoke(std::move(f), std::forward<Args>(args)...)))
{
return !std::invoke(std::move(f), std::forward<Args>(args)...);
}
template<class... Args,
std::enable_if_t<negate_invocable_v<const F, Args...>, int> = 0>
constexpr decltype(auto) operator()(Args&&... args) const&&
noexcept(noexcept(!std::invoke(std::move(f), std::forward<Args>(args)...)))
{
return !std::invoke(std::move(f), std::forward<Args>(args)...);
}
// Удалённые перегрузки необходимы начиная с C++20 для предотвращения
// выбора неэквивалентной, но правильно сформированной перегрузки.
template<class... Args,
std::enable_if_t<!negate_invocable_v<F&, Args...>, int> = 0>
void operator()(Args&&...) & = delete;
template<class... Args,
std::enable_if_t<!negate_invocable_v<const F&, Args...>, int> = 0>
void operator()(Args&&...) const& = delete;
template<class... Args,
std::enable_if_t<!negate_invocable_v<F, Args...>, int> = 0>
void operator()(Args&&...) && = delete;
template<class... Args,
std::enable_if_t<!negate_invocable_v<const F, Args...>, int> = 0>
void operator()(Args&&...) const&& = delete;
};
}
template<class F>
constexpr detail::not_fn_t<std::decay_t<F>> not_fn(F&& f)
{
return {std::forward<F>(f)};
}
|
| (2) not_fn |
namespace detail
{
template<auto ConstFn>
struct stateless_not_fn {
template<class... Args>
constexpr auto operator()(Args&&... args) const
noexcept(noexcept(!std::invoke(ConstFn, std::forward<Args>(args)...)))
-> decltype(!std::invoke(ConstFn, std::forward<Args>(args)...))
{
return !std::invoke(ConstFn, std::forward<Args>(args)...);
}
};
}
template<auto ConstFn>
constexpr detail::stateless_not_fn<ConstFn> not_fn() noexcept
{
if constexpr (std::is_pointer_v<decltype(ConstFn)> ||
std::is_member_pointer_v<decltype(ConstFn)>) {
static_assert(ConstFn != nullptr);
}
return {};
}
|
Примечание
std::not_fn предназначен для замены отрицателей std::not1 и std::not2 эпохи C++03.
| Макрос тест функциональности | |||
|---|---|---|---|
__cpp_lib_not_fn |
201603L |
(C++17) | std::not_fn(), (1)
|
202306L |
(C++26) | Разрешает передачу вызываемых объектов в качестве аргументов шаблона, не являющихся типом, в std::not_fn, (2)
|
Пример
#include <cassert>
#include <functional>
bool is_same(int a, int b) noexcept
{
return a == b;
}
struct S
{
int val;
bool is_same(int arg) const noexcept { return val == arg; }
};
int main()
{
// Использование с свободной функцией:
auto is_differ = std::not_fn(is_same);
assert(is_differ(8, 8) == false); // эквивалентно: !is_same(8, 8) == false
assert(is_differ(6, 9) == true); // эквивалентно: !is_same(8, 0) == true
// Использование с функцией-элементом:
auto member_differ = std::not_fn(&S::is_same);
assert(member_differ(S{3}, 3) == false); //: S tmp{6}; !tmp.is_same(6) == false
// noexcept спецификация сохраняется:
static_assert(noexcept(is_differ) == noexcept(is_same));
static_assert(noexcept(member_differ) == noexcept(&S::is_same));
// Использование с функциональным объектом:
auto same = [](int a, int b) { return a == b; };
auto differ = std::not_fn(same);
assert(differ(1, 2) == true); //: !same(1, 2) == true
assert(differ(2, 2) == false); //: !same(2, 2) == false
#if __cpp_lib_not_fn >= 202306L
auto is_differ_cpp26 = std::not_fn<is_same>();
assert(is_differ_cpp26(8, 8) == false);
assert(is_differ_cpp26(6, 9) == true);
auto member_differ_cpp26 = std::not_fn<&S::is_same>();
assert(member_differ_cpp26(S{3}, 3) == false);
auto differ_cpp26 = std::not_fn<same>();
static_assert(differ_cpp26(1, 2) == true);
static_assert(differ_cpp26(2, 2) == false);
#endif
}
Смотрите также
(устарело в C++17)(удалено в C++20) |
создаёт пользовательский объект std::unary_negate (шаблон функции) |
(устарело в C++17)(удалено в C++20) |
создаёт пользовательский объект std::binary_negate (шаблон функции) |