在C語言中,函數名可以當做函數指針傳遞給形參從而實現回調
1
2
3
4
5
6
7
8
9
10
|
void f1() { printf( "f1()\n" ); } void f2() { printf( "f2()\n" ); } void f3() { printf( "f3()\n" ); } void do_func( void (*f)()) { f(); } int main() { do_func(f1); do_func(f2); do_func(f3); } |
在C++11中,實現回調還可以通過函數模板和lambda表達式的方式
1
2
3
4
5
6
7
8
|
template <typename Func> void do_func(Func f) { f(); } int main() { do_func([]() { printf( "f1()" ); }); do_func([]() { printf( "f2()" ); }); do_func([]() { printf( "f3()" ); }); } |
而假如回調函數的代碼實現較為復雜,且具有重用價值,lambda表達式這種一次性的方案就不太適合,在C++11之前,是通過函數對象來實現的。函數對象說白了就是一個類的普通對象,只不過C++可以重載括號運算符,導致調用類的對象的operator()方法時,就像調用函數一樣自然。
而分析本質,其實回調函數就是一種函數簽名(若干個輸入參數、一個輸出參數)的規范,java雖不存在函數聲明,但是java可以用接口來強制規范。
1
2
3
|
interface Funcable { void Func(); } |
這樣只要實現了該接口的類,都有一個函數簽名和void Func()一致的成員函數(嘛,還是不習慣方法(method)這種叫法),于是只需要把實現了該接口的類的對象傳入函數中,然后在函數中調用該對象的Func()方法即可
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
class F1 implements Funcable { @Override public void Func() { System.out.println( "f1()" ); } } public class Test { public static void do_func(Funcable funcable) { funcable.Func(); } public static void main(String[] args) { do_func( new F1()); } } |
這里節省代碼量,就不把類F2、F3給寫出來了。并且利用java的匿名類可以節省代碼,類似于lambda表達式
1
2
3
4
5
6
|
do_func( new Funcable() { @Override public void Func() { System.out.println( "f2()" ); } }); |
說到lambda表達式,它是可以捕獲外部變量的,在Java這種方式還可以通過匿名內的匿名構造函數來顯式捕獲外部的變量
1
2
3
4
5
6
7
8
9
10
11
|
String msg = "f3()" ; do_func( new Funcable() { String _msg; { _msg = msg; } @Override public void Func() { System.out.println(_msg); } }); |
這種做法就很像lambda表達式了,因為匿名類的匿名構造函數是只能以外部變量為構造參數的,相當于lambda表達式的“捕獲”,對應C++的lambda表達式寫法就是
1
2
|
std::string msg = "f3()" ; do_func([&msg]() { std::cout << msg << std::endl; }); |
java8也有lambda表達式了,因此可以寫成這樣
do_func(() -> { System.out.println(msg); });
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作能帶來一定的幫助~如果有疑問大家可以留言交流,謝謝大家對服務器之家的支持!
原文鏈接:http://www.mamicode.com/info-detail-1967633.html