這篇文章主要介紹了Java透過匿名類別來實現回呼函數的例子,回呼函數就是一種函數簽章(若干個輸入參數、一個輸出參數)的規範,java雖然不存在函數聲明,但是java可以用接口來強制規範。具體操作步驟大家可查看下文的詳細講解,有興趣的朋友可以參考一下。
在C語言中,函數名稱可以當做函數指標傳遞給形參從而實現回呼
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表達式的方式
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可以用介面來強制規範。
interface Funcable { void Func(); }
這樣只要實作了該介面的類,都有一個函數簽名和void Func()一致的成員函數(嘛,還是不習慣方法(method)這種叫法),於是只需要把實作了該介面的類別的物件傳入函數中,然後在函數中呼叫該物件的Func()方法即可
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表達式
do_func(new Funcable() { @Override public void Func() { System.out.println("f2()"); } });
說到lambda表達式,它是可以捕獲外部變數的,在Java這種方式也可以透過匿名內的匿名建構函數來明確地捕捉外部的變數
String msg = "f3()"; do_func(new Funcable() { String _msg; { _msg = msg; } @Override public void Func() { System.out.println(_msg); } });
這種做法就很像lambda表達式了,因為匿名類別的匿名建構函數是只能以外部變數為建構參數的,相當於lambda表達式的“捕獲”,對應C++的lambda表達式寫法就是
std::string msg = "f3()"; do_func([&msg]() { std::cout << msg << std::endl; });
java8也有lambda表達式了,因此可以寫成這樣
do_func(() -> { System.out.println(msg); });
以上是Java使用匿名類別實作回呼函數的範例講解的詳細內容。更多資訊請關注PHP中文網其他相關文章!