深入淺出Java回撥機制
前幾天看了一下Spring的部分原始碼,發現回撥機制被大量使用,覺得有必要把Java回撥機制的理解歸納總結一下,以方便在研究類似於Spring原始碼這樣的程式碼時能更加得心應手。
注:本文不想扯很多拗口的話來充場面,我的目的是希望以最簡明扼要的語言將Java回撥的大概機制說清楚。好了,言歸正傳。
一句話,回撥是一種雙向呼叫模式,什麼意思呢,就是說,被呼叫方在被呼叫時也會呼叫對方,這就叫回撥。“If you call me, i will call back”。
不理解?沒關係,先看看這個可以說比較經典的使用回撥的方式:
是不是清晰一點了?下面再來看一個完全符合這個方式模板的例子
(PS:這個例子來源於網路,由於這個例子表現的功能極度拉風,令我感覺想想出一個超越它的例子確實比較困難,所以直接搬過來)
現在應該對回撥有一點概念了吧。
可是問題來了,對於上面這個例子來說,看不出用回撥有什麼好處,直接在呼叫方法不就可以了,為什麼要使用回撥呢?
事實上,很多需要進行回撥的操作是比較費時的,被呼叫者進行費時操作,然後操作完之後將結果回撥給呼叫者。看這樣一個例子:
可能上面這個例子你不能一眼看出個所以然來,因為其實這裡A是作為一個內部匿名類存在的。好,不要急,讓我們把這個例子來重構一下:
好了,現在就明白多了吧,完全可以轉化為上面所說的回撥使用方式的模板。
現在在來看看為什麼要使用回撥,取得連線getConnection();是費時操作,A希望由B來進行這個費時的操作,執行完了之後通知A即可(即所謂的i call you back)。這就是這裡使用回撥的原因。
在網上看到了一個比喻,覺得很形象,這裡借用一下:
你有一個複雜的問題解決不了,打電話給你的同學,你的同學說可以解決這個問題,但是需要一些時間,那麼你不可能一直拿著電話在那裡等,你會把你的電話號碼告訴他,讓他解決之後打電話通知你。回撥就是體現在你的同學又反過來撥打你的號碼。
結合到前面所分析的,你打電話給你同學就是【you call me】,你同學解決完之後打電話給你就是【i call you back】。
怎麼樣,現在理解了吧?
---------------------------------以下為更新----------------------------------
看了有些朋友的回帖,我又思考了一下,感覺自己之前對回撥作用的理解的確存在偏差。
下面把自己整理之後的想法共享一下,如果有錯誤希望指出!多謝!
先說上面這段程式碼,本來完全可以用模板模式來進行實現:
如果改為回撥實現是這樣的:
可見摒棄了繼承抽象類方式的回撥方式更加簡便靈活。不需要為了實現抽象方法而總是繼承抽象類,而是隻需要通過回撥來增加一個方法即可,更加的直觀簡潔靈活。這算是回撥的好處之一。
下面再給出一個關於利用回撥配合非同步呼叫的很不錯的例子,來源於http://kt8668.iteye.com/blog/205739
回撥介面:
訊息的傳送者:
訊息的接收者:
由上面這個例子可見,回撥可以作為非同步呼叫的基礎來實現非同步呼叫。
注:本文不想扯很多拗口的話來充場面,我的目的是希望以最簡明扼要的語言將Java回撥的大概機制說清楚。好了,言歸正傳。
一句話,回撥是一種雙向呼叫模式,什麼意思呢,就是說,被呼叫方在被呼叫時也會呼叫對方,這就叫回撥。“If you call me, i will call back”。
不理解?沒關係,先看看這個可以說比較經典的使用回撥的方式:
- class A實現介面InA ——背景1
- class A中包含一個class B的引用b ——背景2
- class B有一個引數為InA的方法test(InA a) ——背景3
- A的物件a呼叫B的方法傳入自己,test(a) ——這一步相當於you call me
- 然後b就可以在test方法中呼叫InA的方法 ——這一步相當於i call you back
是不是清晰一點了?下面再來看一個完全符合這個方式模板的例子
(PS:這個例子來源於網路,由於這個例子表現的功能極度拉風,令我感覺想想出一個超越它的例子確實比較困難,所以直接搬過來)
- //相當於介面InA
- public interface BoomWTC{
- //獲得拉登的決定
- public benLaDengDecide();
- // 執行轟炸世貿
- public void boom();
- }
- //相當於class A
- public class At$911 implements BoomWTC{//相當於【背景1】
- private boolean decide;
- private TerroristAttack ta;//相當於【背景2】
- public At$911(){
- Date now=new Date();
- SimpleDateFormat myFmt1=new SimpleDateFormat("yy/MM/dd HH:mm");
- this.dicede= myFmt.format(dt).equals("01/09/11 09:44");
- this.ta=new TerroristAttack();
- }
- //獲得拉登的決定
- public boolean benLaDengDecide(){
- return decide;
- }
- // 執行轟炸世貿
- public void boom(){
- ta.attack(new At$911);//class A呼叫class B的方法傳入自己的物件,相當於【you call me】
- }
- }
- //相當於class B
- public class TerroristAttack{
- public TerroristAttack(){
- }
- public attack(BoomWTC bmw){——這相當於【背景3】
- if(bmw.benLaDengDecide()){//class B在方法中回撥class A的方法,相當於【i call you back】
- //let's go.........
- }
- }
- }
現在應該對回撥有一點概念了吧。
可是問題來了,對於上面這個例子來說,看不出用回撥有什麼好處,直接在呼叫方法不就可以了,為什麼要使用回撥呢?
事實上,很多需要進行回撥的操作是比較費時的,被呼叫者進行費時操作,然後操作完之後將結果回撥給呼叫者。看這樣一個例子:
- //模擬Spring中HibernateTemplate回撥機制的程式碼
- interface CallBack{
- public void doCRUD();
- }
- public class HibernateTemplate {
- public void execute(CallBack action){
- getConnection();
- action.doCRUD();
- releaseConnection();
- }
- public void add(){
- execute(new CallBack(){
- public void doCRUD(){
- System.out.println("執行add操作...");
- }
- });
- }
- public void getConnection(){
- System.out.println("獲得連線...");
- }
- public void releaseConnection(){
- System.out.println("釋放連線...");
- }
- }
可能上面這個例子你不能一眼看出個所以然來,因為其實這裡A是作為一個內部匿名類存在的。好,不要急,讓我們把這個例子來重構一下:
- interface CallBack{ //相當於介面InA
- public void doCRUD();
- }
- public class A implements CallBack{//【背景1】
- private B b;//【背景2】
- public void doCRUD(){
- System.out.println("執行add操作...");
- }
- public void add(){
- b.execute(new A());//【you call me】
- }
- }
- public class B{
- public void execute(CallBack action){ //【背景3】
- getConnection();
- action.doCRUD(); //【i call you back】
- releaseConnection();
- }
- public void getConnection(){
- System.out.println("獲得連線...");
- }
- public void releaseConnection(){
- System.out.println("釋放連線...");
- }
- }
好了,現在就明白多了吧,完全可以轉化為上面所說的回撥使用方式的模板。
現在在來看看為什麼要使用回撥,取得連線getConnection();是費時操作,A希望由B來進行這個費時的操作,執行完了之後通知A即可(即所謂的i call you back)。這就是這裡使用回撥的原因。
在網上看到了一個比喻,覺得很形象,這裡借用一下:
你有一個複雜的問題解決不了,打電話給你的同學,你的同學說可以解決這個問題,但是需要一些時間,那麼你不可能一直拿著電話在那裡等,你會把你的電話號碼告訴他,讓他解決之後打電話通知你。回撥就是體現在你的同學又反過來撥打你的號碼。
結合到前面所分析的,你打電話給你同學就是【you call me】,你同學解決完之後打電話給你就是【i call you back】。
怎麼樣,現在理解了吧?
---------------------------------以下為更新----------------------------------
看了有些朋友的回帖,我又思考了一下,感覺自己之前對回撥作用的理解的確存在偏差。
下面把自己整理之後的想法共享一下,如果有錯誤希望指出!多謝!
先說上面這段程式碼,本來完全可以用模板模式來進行實現:
- public abstract class B{
- public void execute(){
- getConnection();
- doCRUD();
- releaseConnection();
- }
- public abstract void doCRUD();
- public void getConnection(){
- System.out.println("獲得連線...");
- }
- public void releaseConnection(){
- System.out.println("釋放連線...");
- }
- }
- public class A extends B{
- public void doCRUD(){
- System.out.println("執行add操作...");
- }
- public void add(){
- doCRUD();
- }
- }
- public class C extends B{
- public void doCRUD(){
- System.out.println("執行delete操作...");
- }
- public void delete(){
- doCRUD();
- }
- }
如果改為回撥實現是這樣的:
- interface CallBack{
- public void doCRUD();
- }
- public class HibernateTemplate {
- public void execute(CallBack action){
- getConnection();
- action.doCRUD();
- releaseConnection();
- }
- public void add(){
- execute(new CallBack(){
- public void doCRUD(){
- System.out.println("執行add操作...");
- }
- });
- }
- public void delete(){
- execute(new CallBack(){
- public void doCRUD(){
- System.out.println("執行delete操作...");
- }
- });
- }
- public void getConnection(){
- System.out.println("獲得連線...");
- }
- public void releaseConnection(){
- System.out.println("釋放連線...");
- }
- }
可見摒棄了繼承抽象類方式的回撥方式更加簡便靈活。不需要為了實現抽象方法而總是繼承抽象類,而是隻需要通過回撥來增加一個方法即可,更加的直觀簡潔靈活。這算是回撥的好處之一。
下面再給出一個關於利用回撥配合非同步呼叫的很不錯的例子,來源於http://kt8668.iteye.com/blog/205739
回撥介面:
- public interface CallBack {
- /**
- * 執行回撥方法
- * @param objects 將處理後的結果作為引數返回給回撥方法
- */
- public void execute(Object... objects );
- }
訊息的傳送者:
- /**
- * 這個類相當於你自己
- */
- public class Local implements CallBack,Runnable{
- private Remote remote;
- /**
- * 傳送出去的訊息
- */
- private String message;
- public Local(Remote remote, String message) {
- super();
- this.remote = remote;
- this.message = message;
- }
- /**
- * 傳送訊息
- */
- public void sendMessage()
- {
- /**當前執行緒的名稱**/
- System.out.println(Thread.currentThread().getName());
- /**建立一個新的執行緒傳送訊息**/
- Thread thread = new Thread(this);
- thread.start();
- /**當前執行緒繼續執行**/
- System.out.println("Message has been sent by Local~!");
- }
- /**
- * 傳送訊息後的回撥函式
- */
- public void execute(Object... objects ) {
- /**列印返回的訊息**/
- System.out.println(objects[0]);
- /**列印傳送訊息的執行緒名稱**/
- System.out.println(Thread.currentThread().getName());
- /**中斷髮送訊息的執行緒**/
- Thread.interrupted();
- }
- public static void main(String[] args)
- {
- Local local = new Local(new Remote(),"Hello");
- local.sendMessage();
- }
- public void run() {
- remote.executeMessage(message, this); //這相當於給同學打電話,打完電話之後,這個執行緒就可以去做其他事情了,只不過等到你的同學打回電話給你的時候你要做出響應
- }
- }
訊息的接收者:
- /**
- * 這個類相當於你的同學
- */
- public class Remote {
- /**
- * 處理訊息
- * @param msg 接收的訊息
- * @param callBack 回撥函式處理類
- */
- public void executeMessage(String msg,CallBack callBack)
- {
- /**模擬遠端類正在處理其他事情,可能需要花費許多時間**/
- for(int i=0;i<1000000000;i++)
- {
- }
- /**處理完其他事情,現在來處理訊息**/
- System.out.println(msg);
- System.out.println("I hava executed the message by Local");
- /**執行回撥**/
- callBack.execute(new String[]{"Nice to meet you~!"}); //這相當於同學執行完之後打電話給你
- }
- }
由上面這個例子可見,回撥可以作為非同步呼叫的基礎來實現非同步呼叫。
相關文章
- 深入淺出: Java回撥機制(非同步)Java非同步
- Java——回撥機制Java
- java回撥函式機制Java函式
- Java回撥機制解讀Java
- Java 回撥機制(CallBack) 趣解Java
- Java介面回撥機制詳解Java
- 夯實Java基礎系列11:深入理解Java中的回撥機制Java
- JVM深入淺出 -- Java記憶體分配機制JVMJava記憶體
- JAVA重試機制多種方式深入淺出Java
- Java 非同步回撥機制例項解析Java非同步
- Java高階篇——深入淺出Java類載入機制Java
- 回撥機制詳解
- 深入淺出JavaScript執行機制JavaScript
- android Binder機制深入淺出Android
- java閉包和回撥淺析Java
- 深入淺出OAuth 2.0授權機制OAuth
- 淺析回撥機制,這是一篇騷騷的文章
- 深入淺出 Runtime(三):訊息機制
- 深入淺出瀏覽器快取機制瀏覽器快取
- 1.3 - Laravel 5.6 - Callbacks 回撥函式機制Laravel函式
- java介面回撥Java
- 深入淺出java的MapJava
- VSCode For Web 深入淺出 -- 外掛載入機制VSCodeWeb
- jni回撥java方法Java
- Java--回撥模型Java模型
- java回撥函式Java函式
- 深入淺出 WebRTC AEC(聲學回聲消除)Web
- 深入淺出JVM(三)之HotSpot虛擬機器類載入機制JVMHotSpot虛擬機
- Java 同步機制淺談Java
- 深入淺出學Java-HashMapJavaHashMap
- 深入淺出 Java 同步器Java
- 深入理解 JavaScript 回撥函式JavaScript函式
- 一個Demo帶你理解Android介面回撥機制Android
- java回撥函式-非同步回撥-簡明講解Java函式非同步
- java 介面回撥經典案例--網路請求回撥Java
- 深入淺出FE(十四)深入淺出websocketWeb
- 深入淺出說強制型別轉換型別
- java 回撥函式示例Java函式