深入浅出Java回调机制

前几天看了一下Spring的部分源码,发现回调机制被大量使用,觉得有必要把Java回调机制的理解归纳总结一下,以方便在研究类似于Spring源码这样的代码时能更加得心应手。 

注:本文不想扯很多拗口的话来充场面,我的目的是希望以最简明扼要的语言将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:这个例子来源于网络,由于这个例子表现的功能极度拉风,令我感觉想想出一个超越它的例子确实比较困难,所以直接搬过来) 

Java代码  收藏代码
  1. //相当于接口InA  
  2. public interface BoomWTC{  
  3.   //获得拉登的决定  
  4.   public benLaDengDecide();  
  5.   
  6.   // 执行轰炸世贸  
  7.   public void boom();  
  8. }  
  9.   
  10. //相当于class A  
  11. public class At$911 implements BoomWTC{//相当于【背景1】  
  12.   private boolean decide;  
  13.   private TerroristAttack ta;//相当于【背景2】  
  14.   
  15.   public At$911(){  
  16.     Date now=new Date();  
  17.     SimpleDateFormat myFmt1=new SimpleDateFormat("yy/MM/dd HH:mm");  
  18.     this.dicede= myFmt.format(dt).equals("01/09/11 09:44");  
  19.     this.ta=new TerroristAttack();  
  20.   }  
  21.   
  22.   //获得拉登的决定  
  23.   public boolean benLaDengDecide(){  
  24.     return decide;  
  25.   }  
  26.   
  27.   // 执行轰炸世贸  
  28.   public void boom(){  
  29.     ta.attack(new At$911);//class A调用class B的方法传入自己的对象,相当于【you call me】  
  30.   }  
  31. }  
  32.   
  33. //相当于class B  
  34. public class TerroristAttack{  
  35.   public TerroristAttack(){  
  36.   }  
  37.   
  38.   public attack(BoomWTC bmw){——这相当于【背景3】  
  39.     if(bmw.benLaDengDecide()){//class B在方法中回调class A的方法,相当于【i call you back】  
  40.      //let's go.........  
  41.     }  
  42.   }  
  43. }  


现在应该对回调有一点概念了吧。 
可是问题来了,对于上面这个例子来说,看不出用回调有什么好处,直接在调用方法不就可以了,为什么要使用回调呢? 
事实上,很多需要进行回调的操作是比较费时的,被调用者进行费时操作,然后操作完之后将结果回调给调用者。看这样一个例子: 

Java代码  收藏代码
  1. //模拟Spring中HibernateTemplate回调机制的代码  
  2.     interface CallBack{     
  3.         public void doCRUD();     
  4.     }    
  5.         
  6.     public class HibernateTemplate {     
  7.             
  8.         public void execute(CallBack action){    
  9.             getConnection();    
  10.             action.doCRUD();    
  11.             releaseConnection();    
  12.         }    
  13.          
  14.         public void add(){    
  15.              execute(new CallBack(){    
  16.                 public void doCRUD(){    
  17.                     System.out.println("执行add操作...");    
  18.                 }    
  19.              });    
  20.         }     
  21.         
  22.         public void getConnection(){    
  23.             System.out.println("获得连接...");    
  24.         }    
  25.             
  26.         public void releaseConnection(){    
  27.             System.out.println("释放连接...");    
  28.         }    
  29.             
  30.     }    


可能上面这个例子你不能一眼看出个所以然来,因为其实这里A是作为一个内部匿名类存在的。好,不要急,让我们把这个例子来重构一下: 

Java代码  收藏代码
  1. interface CallBack{   //相当于接口InA  
  2.     public void doCRUD();     
  3. }    
  4.   
  5. public class A implements CallBack{//【背景1】  
  6.     private B b;//【背景2】  
  7.     public void doCRUD(){    
  8.           System.out.println("执行add操作...");    
  9.      }    
  10.   
  11.      public void add(){    
  12.              b.execute(new A());//【you call me】    
  13.         }    
  14. }  
  15.   
  16. public class B{  
  17.      public void execute(CallBack action){  //【背景3】  
  18.             getConnection();    
  19.             action.doCRUD();  //【i call you back】  
  20.             releaseConnection();    
  21.         }    
  22.   
  23.       public void getConnection(){    
  24.             System.out.println("获得连接...");    
  25.         }    
  26.             
  27.         public void releaseConnection(){    
  28.             System.out.println("释放连接...");    
  29.         }    
  30. }  


好了,现在就明白多了吧,完全可以转化为上面所说的回调使用方式的模板。 
现在在来看看为什么要使用回调,取得连接getConnection();是费时操作,A希望由B来进行这个费时的操作,执行完了之后通知A即可(即所谓的i call you back)。这就是这里使用回调的原因。 

在网上看到了一个比喻,觉得很形象,这里借用一下: 
你有一个复杂的问题解决不了,打电话给你的同学,你的同学说可以解决这个问题,但是需要一些时间,那么你不可能一直拿着电话在那里等,你会把你的电话号码告诉他,让他解决之后打电话通知你。回调就是体现在你的同学又反过来拨打你的号码。 
结合到前面所分析的,你打电话给你同学就是【you call me】,你同学解决完之后打电话给你就是【i call you back】。 

怎么样,现在理解了吧?  

---------------------------------以下为更新---------------------------------- 

看了有些朋友的回帖,我又思考了一下,感觉自己之前对回调作用的理解的确存在偏差。 
下面把自己整理之后的想法共享一下,如果有错误希望指出!多谢! 

先说上面这段代码,本来完全可以用模板模式来进行实现: 

Java代码  收藏代码
  1. public abstract class B{  
  2.      public void execute(){   
  3.             getConnection();    
  4.             doCRUD();    
  5.             releaseConnection();    
  6.         }    
  7.   
  8.       public abstract void doCRUD();  
  9.   
  10.       public void getConnection(){    
  11.             System.out.println("获得连接...");    
  12.         }    
  13.             
  14.         public void releaseConnection(){    
  15.             System.out.println("释放连接...");    
  16.         }    
  17. }  
  18.   
  19. public class A extends B{  
  20.     public void doCRUD(){    
  21.           System.out.println("执行add操作...");    
  22.      }    
  23.   
  24.      public void add(){    
  25.              doCRUD();  
  26.         }    
  27. }  
  28.   
  29. public class C extends B{  
  30.     public void doCRUD(){    
  31.           System.out.println("执行delete操作...");    
  32.      }    
  33.   
  34.      public void delete(){    
  35.              doCRUD();  
  36.         }    
  37. }  


如果改为回调实现是这样的: 

Java代码  收藏代码
  1. interface CallBack{     
  2.     public void doCRUD();     
  3. }    
  4.     
  5. public class HibernateTemplate {     
  6.     public void execute(CallBack action){    
  7.         getConnection();    
  8.         action.doCRUD();    
  9.         releaseConnection();    
  10.     }    
  11.      
  12.     public void add(){    
  13.          execute(new CallBack(){    
  14.             public void doCRUD(){    
  15.                 System.out.println("执行add操作...");    
  16.             }    
  17.          });    
  18.      }     
  19.   
  20.      public void delete(){    
  21.          execute(new CallBack(){    
  22.             public void doCRUD(){    
  23.                 System.out.println("执行delete操作...");    
  24.             }    
  25.          });    
  26.      }   
  27.     
  28.     public void getConnection(){    
  29.         System.out.println("获得连接...");    
  30.     }    
  31.         
  32.     public void releaseConnection(){    
  33.         System.out.println("释放连接...");    
  34.     }    
  35.         
  36. }    


可见摒弃了继承抽象类方式的回调方式更加简便灵活。不需要为了实现抽象方法而总是继承抽象类,而是只需要通过回调来增加一个方法即可,更加的直观简洁灵活。这算是回调的好处之一。 

下面再给出一个关于利用回调配合异步调用的很不错的例子 
回调接口: 

Java代码  收藏代码
  1. public interface CallBack {    
  2.     /**  
  3.      * 执行回调方法  
  4.      * @param objects   将处理后的结果作为参数返回给回调方法  
  5.      */    
  6.     public void execute(Object... objects );    
  7. }    


消息的发送者: 

Java代码  收藏代码
  1. /** 
  2.  * 这个类相当于你自己 
  3.  */  
  4. public class Local implements CallBack,Runnable{    
  5.      
  6.     private Remote remote;    
  7.         
  8.     /**  
  9.      * 发送出去的消息  
  10.      */    
  11.     private String message;    
  12.         
  13.     public Local(Remote remote, String message) {    
  14.         super();    
  15.         this.remote = remote;    
  16.         this.message = message;    
  17.     }    
  18.     
  19.     /**  
  20.      * 发送消息  
  21.      */    
  22.     public void sendMessage()    
  23.     {    
  24.         /**当前线程的名称**/    
  25.         System.out.println(Thread.currentThread().getName());    
  26.         /**创建一个新的线程发送消息**/    
  27.         Thread thread = new Thread(this);    
  28.         thread.start();    
  29.         /**当前线程继续执行**/    
  30.         System.out.println("Message has been sent by Local~!");    
  31.     }    
  32.     
  33.     /**  
  34.      * 发送消息后的回调函数  
  35.      */    
  36.     public void execute(Object... objects ) {    
  37.         /**打印返回的消息**/    
  38.         System.out.println(objects[0]);    
  39.         /**打印发送消息的线程名称**/    
  40.         System.out.println(Thread.currentThread().getName());    
  41.         /**中断发送消息的线程**/    
  42.         Thread.interrupted();    
  43.     }    
  44.         
  45.     public static void main(String[] args)    
  46.     {    
  47.         Local local = new Local(new Remote(),"Hello");    
  48.             
  49.         local.sendMessage();    
  50.     }    
  51.     
  52.     public void run() {    
  53.         remote.executeMessage(message, this);  //这相当于给同学打电话,打完电话之后,这个线程就可以去做其他事情了,只不过等到你的同学打回电话给你的时候你要做出响应  
  54.             
  55.     }    
  56. }    


消息的接收者: 

Java代码  收藏代码
  1. /** 
  2.  * 这个类相当于你的同学 
  3.  */  
  4. public class Remote {    
  5.     
  6.     /**  
  7.      * 处理消息  
  8.      * @param msg   接收的消息  
  9.      * @param callBack  回调函数处理类  
  10.      */    
  11.     public void executeMessage(String msg,CallBack callBack)    
  12.     {    
  13.         /**模拟远程类正在处理其他事情,可能需要花费许多时间**/    
  14.         for(int i=0;i<1000000000;i++)    
  15.         {    
  16.                 
  17.         }    
  18.         /**处理完其他事情,现在来处理消息**/    
  19.         System.out.println(msg);    
  20.         System.out.println("I hava executed the message by Local");    
  21.         /**执行回调**/    
  22.         callBack.execute(new String[]{"Nice to meet you~!"});  //这相当于同学执行完之后打电话给你  
  23.     }    
  24.         
  25. }    


由上面这个例子可见,回调可以作为异步调用的基础来实现异步调用。

转载于:https://www.cnblogs.com/smilesmile/p/3833385.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/473257.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

前端:实现div等块元素添加X轴滚动显示(Y轴不滚动)

一、建立外盒子与内盒子 原生态代码&#xff1a; <div class"tol_dev"><div class"dev_li"></div><div class"dev_li"></div><div class"dev_li"></div><div class"dev_li"…

2020年学习总结

文章目录1. CSDN 博客数据2. 基础算法练习3. 机器学习4. 深度学习5. MySQL6. 总结和展望时间过得很快&#xff0c;2020结束了&#xff01; 写个流水账&#xff0c;记录一下。 1. CSDN 博客数据 截个图对比下&#xff1a; 2019年终2020年终 2. 基础算法练习 LeetCode 刷题 …

npm全局环境变量配置及解决VsCode使用时遇到的问题

一、npm全局环境变量配置 1、我们要先配置npm的全局模块的存放路径以及cache的路径 例如我希望将以上两个文件夹放在NodeJS的主目录下&#xff0c;便在NodeJs下建立”node_global”及”node_cache”两个文件夹。如下图 2、cmd 中输入如下命令 npm config set prefix “d:\no…

Android菜鸟如何学习Android系统开发?

如何做好Android学习前的准备? 如果你已经确定了学习Android的目标&#xff0c;那么&#xff0c;应该提前做好哪些工作、先打下哪些基础呢? 首先&#xff0c;你最好先熟悉一门编程语言&#xff0c;现在大学里面和计算机相关的专业甚至理工类专业一般都会开设C语言课程&#x…

关于Django中JsonResponse返回中文字典编码错误的解决方案

遇到这样一个问题&#xff0c;返回的json不是中文 def get_json(request):return JsonResponse({"res": "成功"}) 结果&#xff1a; {"res": "\u6210\u529f"} 解决方案&#xff1a;JsonResponse(data, json_dumps_params{ensure_a…

LeetCode 668. 乘法表中第k小的数(二分查找)

文章目录1. 题目2. 解题1. 题目 几乎每一个人都用 乘法表。但是你能在乘法表中快速找到第k小的数字吗&#xff1f; 给定高度m 、宽度n 的一张 m * n的乘法表&#xff0c;以及正整数k&#xff0c;你需要返回表中第k 小的数字。 例 1&#xff1a; 输入: m 3, n 3, k 5 输出…

天池 在线编程 寻找比周围都大的点(模拟)

文章目录1. 题目2. 解题1. 题目 https://tianchi.aliyun.com/oj/245679029019779851/254275128279634587 给一个n*m大小的矩阵&#xff0c;寻找矩阵中所有比邻居&#xff08;上下左右&#xff0c;对角也算&#xff0c;不考虑边界就是8个咯&#xff09;都严格大的点。 返回一个…

[原创][R语言]股票分析实战[4]:周级别涨幅趋势的相关性

[简介] 常用网名: 猪头三 出生日期: 1981.XX.XX QQ联系: 643439947 个人网站: 80x86汇编小站 https://www.x86asm.org 编程生涯: 2001年~至今[共22年] 职业生涯: 20年 开发语言: C/C、80x86ASM、PHP、Perl、Objective-C、Object Pascal、C#、Python 开发工具: Visual Studio、D…

前端:实现手机左右滑动效果

需求&#xff1a;手机可以左右滑动&#xff0c;显示商品或者div信息 原理&#xff1a;建立两个盒子&#xff0c;一个是可以看到的手机屏幕盒子&#xff0c;一个是自己设定好的盒子&#xff0c;左右滑动&#xff0c;只显示与手机屏幕盒子交集的部分 代码&#xff1a; <div …

天池 在线编程 双向取数(博弈DP)

文章目录1. 题目2. 解题1. 题目 https://tianchi.aliyun.com/oj/245679029019779851/254275128279634585 有一个长度为n的数列arr&#xff0c; 甲乙两个人每次可以从头或者从末尾取一个数&#xff0c;双方都想让自己取数之和尽量多&#xff0c; 甲先取数&#xff0c;问甲乙在…

前端demo:实现背景半透明,div与文本正常显示

一、需求&#xff1a; 背景实现透明度0.3&#xff0c;其他组件正常显示 二、代码&#xff1a; <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-widt…

天池 在线编程 布尔表达式求值(栈)

文章目录1. 题目2. 解题1. 题目 https://tianchi.aliyun.com/oj/245679029019779851/254275128279634588 给定一个字符串代表一个仅包含"true","false","or","and"的布尔表达式。 你的任务是将这个表达式的值求出&#xff0c;返回&q…

Schema详解

XML Schema 简介XSD 为何使用XML Schema 是基于 XML 的 DTD 替代者。XML Schema 可描述 XML 文档的结构。XML Schema 语言也可作为 XSD&#xff08;XML Schema Definition&#xff09;来引用。在继续学习之前&#xff0c;您需要对下面的知识有基本的了解&#xff1a;HTML / XHT…

整理总结一下:git恢复本地误删除的分支

一、使用git log -g查看日志&#xff0c;找回之前提交的commit,并记下commit_id git log -g 二、新建分支newbranch&#xff0c;把commit_id分支复制到新的分支上 git branch newbranch commit_id 三、切换新建分支newbranch&#xff0c;检查文件 git checkout newbran…

LeetCode 1074. 元素和为目标值的子矩阵数量(2d前缀和+哈希)

文章目录1. 题目2. 解题1. 题目 给出矩阵 matrix 和目标值 target&#xff0c;返回元素总和等于目标值的非空子矩阵的数量。 子矩阵 x1, y1, x2, y2 是满足 x1 < x < x2 且 y1 < y < y2 的所有单元 matrix[x][y] 的集合。 如果 (x1, y1, x2, y2) 和 (x1, y1, x2…

最炫国漫《雾山五行》用 Python 了解一下到底有多优秀

看动漫的小伙伴应该知道最近出了一部神漫《雾山五行》&#xff1a; 1、极具特色的水墨画风和超燃的打斗场面广受好评 2、首集播出不到 24 小时登顶 B 站热搜第一&#xff0c;豆瓣开分 9.5&#xff0c; 火爆程度可见一斑&#xff0c;就打斗场面而言&#xff0c;说是最炫动漫也…

LeetCode 982. 按位与为零的三元组(位运算+计数)

文章目录1. 题目2. 解题1. 题目 给定一个整数数组 A&#xff0c;找出索引为 (i, j, k) 的三元组&#xff0c;使得&#xff1a; 0 < i < A.length 0 < j < A.length 0 < k < A.length A[i] & A[j] & A[k] 0&#xff0c;其中 & 表示按位与&…

雷军一往无前的十年(小米十周年公开演讲)附赠《一往无前》电子书籍

▲雷军 | 中国企业家俱乐部理事、小米科技董事长 做全球最好的手机&#xff0c;只卖一半的价钱&#xff0c;让每个人都能买得起——如何能够实现这个看上去、听上去都不靠谱的目标&#xff1f; 来源 | 雷军公众号分享 | 雷军 2020年8月11日19:30&#xff0c;小米十周年&#x…

LeetCode 1147. 段式回文(贪心)

文章目录1. 题目2. 解题1. 题目 段式回文 其实与 一般回文 类似&#xff0c;只不过是最小的单位是 一段字符 而不是 单个字母。 举个例子&#xff0c;对于一般回文 "abcba" 是回文&#xff0c;而 "volvo" 不是&#xff0c;但如果我们把 "volvo"…

利用python批量修改文件名称

一、基础知识&#xff1a; 通过查阅资料os模块中rename和renames都可以做到 他们的区别为.rename:只能修改文件名 renames:可以修改文件名,还可以修改文件上级目录名称 另一个用到的方法是os.listdir(path) path为路径 此方法可以将指定路径文件夹中的文件名录入一个列表…