C++ åèæå
- C++11
- C++14
- C++17
- C++20
- C++ ç¼è¯å¨æ¯ææ åµè¡¨
- ç¬ç«ä¸å®¿ä¸»å®ç°
- C++ è¯è¨
- C++ å ³é®è¯
- é¢å¤çå¨
- C++ æ ååºå¤´æä»¶
- å ·åè¦æ±
- åè½ç¹æ§æµè¯ (C++20)
- å·¥å ·åº
- ç¨åºæ¯æå·¥å ·
- std::initializer_list
- 彿°å¯¹è±¡
- std::less
- std::mem_fn
- std::bind
- std::function
- std::bad_function_call
- std::is_bind_expression
- std::is_placeholder
- std::placeholders::_1, std::placeholders::_2, ..., std::placeholders::_N
- std::invoke
- std::not_fn
- std::bind_front
- std::boyer_moore_searcher
- std::default_searcher
- std::identity
- std::reference_wrapper
- std::ref, std::cref
- std::unwrap_reference, std::unwrap_ref_decay
- std::plus
- std::minus
- std::negate
- std::multiplies
- std::divides
- std::modulus
- std::logical_and
- std::logical_or
- std::logical_not
- std::equal_to
- std::not_equal_to
- std::greater
- std::greater_equal
- std::less_equal
- std::bit_and
- std::bit_or
- std::bit_xor
- std::bit_not
- std::boyer_moore_horspool_searcher
- std::unary_function
- std::binary_function
- std::ptr_fun
- std::pointer_to_unary_function
- std::pointer_to_binary_function
- std::mem_fun_t, std::mem_fun1_t, std::const_mem_fun_t, std::const_mem_fun1_t
- std::not1
- std::not2
- std::bind1st, std::bind2nd
- std::mem_fun_ref_t, std::mem_fun1_ref_t, std::const_mem_fun_ref_t, std::const_mem_fun1_ref_t
- std::unary_negate
- std::binary_negate
- std::less<void>
- std::apply
- åºç¹æ§æµè¯å® (C++20)
- std::hash
- std::pair
- std::tuple
- std::optional
- std::any
- std::variant
- æ ¼å¼ååº (C++20)
- std::integer_sequence
- std::exchange
- std::make_from_tuple
- std::launder
- std::to_chars
- std::from_chars
- std::as_const
- std::source_location
- å忰彿°
- std::bitset
- std::cmp_equal, cmp_not_equal, cmp_less, cmp_greater, cmp_less_equal, cmp_greater_equal
- std::in_range
- std::declval
- std::forward
- std::move
- std::move_if_noexcept
- std::chars_format
- std::piecewise_construct_t
- std::piecewise_construct
- std::in_place, std::in_place_type, std::in_place_index, std::in_place_t, std::in_place_type_t, std::in_place_index_t
- 注é
- ç±»åæ¯æï¼åºæ¬ç±»åãRTTIãç±»åç¹æ§ï¼
- æ¦å¿µåº (C++20)
- é误å¤ç
- 卿å å管ç
- æ¥æåæ¶é´å·¥å ·
- å符串åº
- 容å¨åº
- è¿ä»£å¨åº
- èå´åº (C++20)
- ç®æ³åº
- æ°å¼åº
- è¾å ¥/è¾åºåº
- æä»¶ç³»ç»åº
- æ¬å°ååº
- æ£å表达å¼åº
- ååæä½åº
- çº¿ç¨æ¯æåº
- å®éªæ§ C++ ç¹æ§
- æç¨çèµæº
- ç´¢å¼
- std 符å·ç´¢å¼
- åç¨æ¯æ (C++20)
- C++ å ³é®è¯
ä½ç½®ï¼é¦é¡µ > C++ åèæå >å·¥å ·åº >彿°å¯¹è±¡ > std::bind
std::bind
彿°æ¨¡æ¿ bind çæ f ç转åè°ç¨å
è£
å¨ãè°ç¨æ¤å
è£
å¨çä»·äºä»¥ä¸äºç»å®å° args çåæ°è°ç¨ f ã
åæ°
| f | - | å¯è°ç¨ (Callable) 对象ï¼å½æ°å¯¹è±¡ãæå彿°æéãå°å½æ°å¼ç¨ãæåæå彿°æéææåæ°æ®æåæéï¼ |
| args | - | è¦ç»å®çåæ°åè¡¨ï¼æªç»å®åæ°ä¸ºå½åç©ºé´ std::placeholders çå ä½ç¬¦ _1, _2, _3... ææ¿æ¢
|
è¿åå¼
æªæå®ç±»å T ç彿°å¯¹è±¡ï¼æ»¡è¶³ std::is_bind_expression<T>::value == true ã宿ä¸å屿§ï¼
std::bind è¿åç±»å
æå对象
std::bind çè¿åç±»åä¿æä» std::forward<F>(f) æé ç std::decay<F>::type ç±»åæå对象ï¼åå¯¹äºæ¯ä¸ª args... ç±»å为 std::decay<Arg_i>::type çåä¸ä¸ªå¯¹è±¡ï¼ç±»ä¼¼å°ä» std::forward<Arg_i>(arg_i) æé ã
æé 彿°
è¥ std::bind çè¿åç±»åçæææåç±»åï¼è¯´æå¦ä¸ï¼ä¸ºå¯å¤å¶æé (CopyConstructible) ï¼åå®ä¸ºå¯å¤å¶æé (CopyConstructible) ï¼å¦å为å¯ç§»å¨æé (MoveConstructible) ãç±»åå®ä¹ä¸åæåï¼
æåç±»å
|
(C++20 å) |
æå彿° operator()
ç»å®ä»å
åå° bind è°ç¨è·å¾ç对象 g ï¼ä»å½æ°è°ç¨è¡¨è¾¾å¼ g(u1, u2, ... uM) è°ç¨å®æ¶ï¼åç被åå¨å¯¹è±¡çè°ç¨ï¼å¦å以 std::invoke(fd, std::forward<V1>(v1), std::forward<V2>(v2), ..., std::forward<VN>(vN)) ï¼å
¶ä¸ fd æ¯ std::decay_t<F> ç±»åå¼ï¼åç»å® v1, v2, ..., vN åæ°çå¼åç±»åæä»¥ä¸æ¹å¼ç¡®å®ã
- è¥åå¨çåæ°
argæ¥æç±»å std::reference_wrapper<T> ï¼ä¾å¦ï¼å¨èµ·å§çå°bindè°ç¨ä¸ä½¿ç¨äº std::ref æ std::cref ï¼ï¼åä¸è¿°std::invokeè°ç¨ä¸çvnæ¯arg.get()ä¸åä¸è°ç¨ä¸çç±»åVnæ¯T&ï¼åå¨çåæ°æå¼ç¨ä¼ éè¿å ¥è¢«è°ç¨ç彿°å¯¹è±¡ã - è¥åå¨çåæ°
argæ¥æç±»åT并满足 std::is_bind_expression<T>::value == true ï¼ä¾å¦ï¼ç´æ¥ä¼ éå°èµ·å§ç对bindè°ç¨çå¦ä¸bind表达å¼ï¼ï¼åbindè¿è¡å½æ°ç»åï¼ä¸æ¯ä¼ é该 bind å表达å¼å°è¿åç彿°å¯¹è±¡ï¼èæ¯é¥¥æ¸´å°è°ç¨è¯¥å表达å¼ï¼å¹¶å°å ¶è¿åå¼ä¼ éç»å¤å±å¯è°ç¨å¯¹è±¡ãè¥ bind åè¡¨è¾¾å¼æ¥æä»»ä½å ä½ç¬¦åæ°ï¼åå°å®ä»¬ä¸å¤å± bind å ±äº«ï¼ä»u1, u2, ...ä¸éåºï¼ãç¹å«æ¯ï¼ä¸è¿°std::invokeè°ç¨ä¸çåæ°vnæ¯arg(std::forward<Uj>(uj)...)èåä¸è°ç¨ä¸çç±»åVnæ¯ std::result_of_t<T cv &(Uj&&...)>&& ï¼ cv éå®ä¸gçç¸åï¼ã - è¥åå¨çåæ°
argæ¥æç±»åT并满足 std::is_placeholder<T>::value != 0 ï¼è¡¨ç¤ºä»¥å¦std::placeholders::_1, _2, _3, ...çå ä½ç¬¦ä¸ºå°bindåå§è°ç¨çåæ°ï¼ï¼åå°å ä½ç¬¦ææç¤ºçåæ°ï¼_1çu1ã_2çu2çï¼ä¼ éç»å¯è°ç¨å¯¹è±¡ï¼ä¸è¿°std::invokeè°ç¨ä¸çåæ°vnæ¯ std::forward<Uj>(uj) èåä¸è°ç¨ä¸å¯¹åºç±»åVnæ¯ Uj&& ã - å¦åï¼æ®éçåå¨åæ°
argä½ä¸ºå·¦å¼åæ°ä¼ éç»ï¼ä¸è¿°std::invokeè°ç¨ä¸çåæ°vnåçº¯å°æ¯argä¸å¯¹åºç±»åVnæ¯T cv &ï¼å ¶ä¸ cv æ¯ä¸gç¸åç cv éå®ã
è¥æä¾äºå° g() è°ç¨çä¸äºåæ°ä¸å¹é
åå¨äº g çä»»ä½å ä½ç¬¦ï¼åæ±å¼å¹¶å¿½ç¥æªä½¿ç¨çåæ°ã
è¥ g 为 volatile éå®ï¼å³å
¶ cv éå®ç¬¦æ¯ volatile æ const volatile ï¼ï¼åè¡ä¸ºæªå®ä¹ã
å¼å¸¸
ä»
è¥ä» std::forward<F>(f) æé std::decay<F>::type æåºï¼æä» std::forward<Arg_i>(arg_i) æé 对åºçä»»ä½ std::decay<Arg_i>::type æåºææåºå¼å¸¸ï¼å
¶ä¸ Arg_i æ¯ Args... args ä¸ç¬¬ i 个类åï¼è arg_i æ¯ç¬¬ i ä¸ªåæ°ã
注解
å¦å¯è°ç¨ (Callable) ä¸æè¿°ï¼è°ç¨æåééææå彿°æéææåééææ°æ®æåæéæ¶ï¼é¦åæ°å¿ é¡»æ¯å¼ç¨ææéï¼å¯ä»¥å 嫿ºè½æéï¼å¦ std::shared_ptr ä¸ std::unique_ptrï¼ï¼æåå°è®¿é®å ¶æåç对象ã
å° bind çåæ°è¢«å¤å¶æç§»å¨ï¼èä¸å³ä¸æå¼ç¨ä¼ éï¼é¤éå è£ äº std::ref æ std::cref ã
å
许åä¸ bind 表达å¼ä¸çå¤éå ä½ç¬¦ï¼ä¾å¦å¤ä¸ª _1 ï¼ï¼ä½ç»æä»
è¥å¯¹åºåæ°ï¼ u1 ï¼æ¯å·¦å¼æä¸å¯ç§»å¨å³å¼æè¯å¥½å®ä¹ã
示ä¾
#include <random> #include <iostream> #include <memory> #include <functional>  void f(int n1, int n2, int n3, const int& n4, int n5) { std::cout << n1 << ' ' << n2 << ' ' << n3 << ' ' << n4 << ' ' << n5 << '\n'; }  int g(int n1) { return n1; }  struct Foo { void print_sum(int n1, int n2) { std::cout << n1+n2 << '\n'; } int data = 10; };  int main() { using namespace std::placeholders; // å¯¹äº _1, _2, _3...  // æ¼ç¤ºåæ°éæåºåæå¼ç¨ä¼ é int n = 7; // ï¼ _1 ä¸ _2 æ¥èª std::placeholders ï¼å¹¶è¡¨ç¤ºå°æ¥ä¼ä¼ éç» f1 çåæ°ï¼ auto f1 = std::bind(f, _2, 42, _1, std::cref(n), n); n = 10; f1(1, 2, 1001); // 1 为 _1 æç»å®ï¼ 2 为 _2 æç»å®ï¼ä¸ä½¿ç¨ 1001 // è¿è¡å° f(2, 42, 1, n, 7) çè°ç¨  // åµå¥ bind å表达å¼å ±äº«å ä½ç¬¦ auto f2 = std::bind(f, _3, std::bind(g, _3), _3, 4, 5); f2(10, 11, 12); // è¿è¡å° f(12, g(12), 12, 4, 5); çè°ç¨  // 常è§ä½¿ç¨æ åµï¼ä»¥åå¸ç»å® RNG std::default_random_engine e; std::uniform_int_distribution<> d(0, 10); std::function<int()> rnd = std::bind(d, e); // e çä¸ä¸ªå¯æ¬åå¨äº rnd for(int n=0; n<10; ++n) std::cout << rnd() << ' '; std::cout << '\n';  // ç»å®æåæå彿°æé Foo foo; auto f3 = std::bind(&Foo::print_sum, &foo, 95, _1); f3(5);  // ç»å®æåæ°æ®æåæé auto f4 = std::bind(&Foo::data, _1); std::cout << f4(foo) << '\n';  // æºè½æé亦è½ç¨äºè°ç¨è¢«å¼ç¨å¯¹è±¡çæå std::cout << f4(std::make_shared<Foo>(foo)) << '\n' << f4(std::make_unique<Foo>(foo)) << '\n'; }
è¾åºï¼
2 42 1 10 7 12 12 12 4 5 1 5 0 2 0 8 2 2 10 8 100 10 10 10
åé
| (C++20) |
æé¡ºåºç»å®ä¸å®æ°éçåæ°å°å½æ°å¯¹è±¡ (彿°æ¨¡æ¿) |
| (C++11) |
ç¨ä½ std::bind 表达å¼ä¸çæªç»å®å®åçå ä½ç¬¦ (常é) |
| (C++11) |
仿åæéå建åºå½æ°å¯¹è±¡ (彿°æ¨¡æ¿) |