std::bind_front, std::bind_back
| Определено в заголовочном файле <functional>
|
||
template< class F, class... Args > constexpr /*не указано*/ bind_front( F&& f, Args&&... args ); |
(1) | (начиная с C++20) |
template< auto f, class... Args > constexpr /* не указано */ bind_front( Args&&... ); |
(2) | (начиная с C++26) |
template< class F, class... Args > constexpr /*не указано*/ bind_back( F&& f, Args&&... args ); |
(3) | (начиная с C++23) |
template< auto f, class... Args > constexpr /* не указано */ bind_back( Args&&... ); |
(4) | (начиная с C++26) |
Шаблоны функций std::bind_front и std::bind_back создают оболочку переадресации вызовов для f. Вызов этой оболочки эквивалентен вызову f с первым (для перегрузок (1,2)) или последним (для перегрузок (3,4)) sizeof...(Args) параметрами, привязанными к args.
Следующие выражения эквивалентны:
std::bind_front(f, bound_args...)(call_args...) и std::invoke(f, bound_args..., call_args...).std::bind_front<f>(bound_args...)(call_args...) и std::invoke(f, bound_args..., call_args...).std::bind_back(f, bound_args...)(call_args...) и std::invoke(f, call_args..., bound_args...).std::bind_back<f>(bound_args...)(call_args...) и std::invoke(f, call_args..., bound_args...).Пусть F это тип f. Программа некорректна, если любое из следующих условий равно false:
std::is_constructible_v<std::decay_t<F>, F>,std::is_move_constructible_v<std::decay_t<F>>,(std::is_pointer_v<F> || std::is_member_pointer_v<F>) == true, тогда f != nullptr,(std::is_constructible_v<std::decay_t<Args>, Args> && ...),(std::is_move_constructible_v<std::decay_t<Args>> && ...).| Этот раздел не завершён Причина: Отразить улучшения C++26 (2,4) из "P2714R1 - привязка спереди и сзади к вызываемым объектам NTTP" |
Параметры
| f | — | Объект Callable (объект функции, указатель на функцию, ссылка на функцию, указатель на функцию-элемент или указатель на элемент данных), который будет привязан к некоторым аргументам |
| args | — | список аргументов для привязки к первому (1,2) или последнему (3,4) параметру sizeof...(Args) объекта f
|
| Требования к типам | ||
- должен соответствовать требованиям MoveConstructible. (1,3)
| ||
- должен соответствовать требованиям MoveConstructible. (1-4)
| ||
Возвращаемое значение
Функциональный объект (оболочка вызова) типа T, не указан, за исключением того, что типы объектов, возвращаемые двумя вызовами std::bind_front или std:: bind_back с теми же аргументами, одинаковы.
Пусть bind-partial будет либо std::bind_front, либо std::bind_back.
Возвращённый объект имеет следующие свойства:
bind-partial return type
Объекты-элементы
The returned object behaves as if it holds:
fd типа std::decay_t<F> инициализированный напрямую не списком из std::forward<F>(f), иtup, созданный с помощью std::tuple<std::decay_t<Args>...>(std::forward<Args>(args)...), за исключением того, что поведение присваивания возвращаемого объекта не указано, а имена предназначены только для пояснения.Конструкторы
Тип возвращаемого значения bind-partial ведёт себя так, как будто его конструкторы копирования/перемещения выполняют копирование/перемещение по элементам. Он CopyConstructible, если все его объекты-элементы (указанные выше) являются CopyConstructible, и MoveConstructible иначе.
Функция-элемент operator()
Учитывая объект G, полученный в результате более раннего вызова (1,3) bind-partial(f, args...) или (2,4) bind-partial<ConstFn>(args...), когда glvalue g, обозначающее G, вызывается в выражение вызова функции g(call_args...), происходит вызов хранимого объекта, как если бы:
std::invoke(g.fd, std::get<Ns>(g.tup)..., call_args...), где bind-partial это std::bind_front,std::invoke(ConstFn, std::get<Ns>(g.tup)..., call_args...), где bind-partial это std::bind_front,std::invoke(g.fd, call_args..., std::get<Ns>(g.tup)...), где bind-partial это std::bind_back,std::invoke(ConstFn, call_args..., std::get<Ns>(g.tup)...), где bind-partial это std::bind_back,где
Nsцелочисленный пакет0, 1, ..., (sizeof...(Args) - 1),gявляется значением lvalue в выражении std::invoke, если оно является значением lvalue в выражении вызова, и значением rvalue иначе. Таким образом,std::move(g)(call_args...)может перемещать связанные аргументы в вызов, гдеg(call_args...)будет копироваться.
Программа некорректна, если g имеет volatile-квалифицированный тип.
Элемент operator() является noexcept, если вызываемое им выражение std::invoke является noexcept (другими словами, он сохраняет спецификацию исключения базового оператора вызова).
Исключения
Может генерировать любое исключение, вызванное вызовом конструктора:
Примечание
Эти шаблоны функций предназначены для замены std::bind. В отличие от std::bind, они не поддерживают произвольную перестановку аргументов и не имеют специальной обработки для вложенных выражений связывания или объектов std::reference_wrapper. С другой стороны, они обращают внимание на категорию значения объекта оболочки вызова и распространяют спецификацию исключений базового оператора вызова.
Как описано в std::invoke, при вызове указателя на нестатическую функцию-элемент или указателя на нестатический элемент данных первый аргумент должен быть ссылкой или указателем (включая, возможно, интеллектуальный указатель например, std::shared_ptr и std::unique_ptr) на объект, элемент которого будет доступен.
Аргументы std::bind_front или std::bind_back копируются или перемещаются и никогда не передаются по ссылке, если они не заключены в std::ref или std::cref.
Обычно привязка аргументов к функции или функции-элементу с использованием (1) std::bind_front и (3) std::bind_back требует сохранения указателя на функцию вместе с аргументами, хотя язык точно знает, какую функцию вызывать, без необходимости разыменовывания указателя. Чтобы гарантировать "нулевую стоимость" в таких случаях, C++26 вводит версии (2,4) (которые принимают вызываемый объект в качестве аргумента для нет названия раздела).
| Макрос тест функциональности | |||
|---|---|---|---|
__cpp_lib_bind_front |
201907L |
(C++20) | std::bind_front, (1)
|
202306L |
(C++26) | Разрешает передачу вызываемых объектов в качестве аргументов шаблона, не являющихся типом, в std::bind_front, (2)
| |
__cpp_lib_bind_back |
202202L |
(C++23) | std::bind_back, (3)
|
202306L |
(C++26) | Разрешает передачу вызываемых объектов в качестве аргументов шаблона, не являющихся типом, в std::bind_back, (4)
|
Пример
#include <cassert>
#include <functional>
int minus(int a, int b)
{
return a - b;
}
struct S
{
int val;
int minus(int arg) const noexcept { return val - arg; }
};
int main()
{
auto fifty_minus = std::bind_front(minus, 50);
assert(fifty_minus(3) == 47); // эквивалентно: minus(50, 3) == 47
auto member_minus = std::bind_front(&S::minus, S{50});
assert(member_minus(3) == 47); //: S tmp{50}; tmp.minus(3) == 47
// noexcept спецификация сохраняется:
static_assert(!noexcept(fifty_minus(3)));
static_assert(noexcept(member_minus(3)));
// Связывание лямбды:
auto plus = [](int a, int b) { return a + b; };
auto forty_plus = std::bind_front(plus, 40);
assert(forty_plus(7) == 47); // эквивалентно: plus(40, 7) == 47
#if __cpp_lib_bind_front >= 202306L
auto fifty_minus_cpp26 = std::bind_front<minus>(50);
assert(fifty_minus_cpp26(3) == 47);
auto member_minus_cpp26 = std::bind_front<&S::minus>(S{50});
assert(member_minus_cpp26(3) == 47);
auto forty_plus_cpp26 = std::bind_front<plus>(40);
assert(forty_plus(7) == 47);
#endif
#if __cpp_lib_bind_back >= 202202L
auto madd = [](int a, int b, int c) { return a * b + c; };
auto mul_plus_seven = std::bind_back(madd, 7);
assert(mul_plus_seven(4, 10) == 47); //: madd(4, 10, 7) == 47
#endif
#if __cpp_lib_bind_back >= 202306L
auto mul_plus_seven_cpp26 = std::bind_back<madd>(7);
assert(mul_plus_seven_cpp26(4, 10) == 47);
#endif
}
Ссылки
- C++26 стандарт (ISO/IEC 14882:2026):
- 22.10.14 Шаблоны функции bind_front and bind_back [func.bind.partial]
- C++23 стандарт (ISO/IEC 14882:2023):
- 22.10.14 Шаблоны функции bind_front and bind_back [func.bind.partial]
- C++20 стандарт (ISO/IEC 14882:2020):
- 20.14.14 Шаблон функции bind_front [func.bind.front]
Смотрите также
(C++11) |
связывает один или несколько аргументов с объектом функцией (шаблон функции) |
(C++11) |
создаёт объект функцию из указателя на элемент (шаблон функции) |