Java回调(callback)机制

回调的核心:回调方将本身即 this 传递给调用方,这样调用方就可以在调用完毕之后告诉回调方想要知道的信息。

一、简述

从软件模块之间的调用方式看,分为三类:同步调用、异步调用和回调。

1️⃣同步调用

同步调用是最基本并且最简单的一种调用方式,类 A 的 a() 调用类 B 的 b(),一直等待 b() 执行完毕,a() 继续往下走。该调用方式适用于 b() 执行时间不长的情况,因为 b() 执行时间过长或者直接阻塞的话,a() 的余下代码是无法执行下去的,这样会造成整个流程的阻塞。

2️⃣异步调用

异步调用是为了解决同步调用可能出现阻塞,导致整个流程卡住而产生的一种调用方式。类 A 的 a() 通过新起线程的方式调用类 B 的 b(),代码接着直接往下执行,这样无论 b() 执行时间多久,都不会阻塞住 a() 的执行。但是,由于 a() 不等待 b() 的执行完成,在 a() 需要 b() 执行结果的情况下(视具体业务而定,有些业务比如启异步线程发个微信通知、刷新缓存这种就没必要),必须通过一定的方式对 b() 的执行结果进行监听。Java 中,可以使用 Future+Callable 的方式做到这一点。

3️⃣回调

在面向对象的语言中,回调则是通过接口或抽象类来实现的,实现这种接口的类为回调类,回调类的对象为回调对象。回调的思想是:

  1. class A 实现接口 CallBack——背景 1
  2. class A 中包含一个 class B 的引用 b——背景 2
  3. class B 有一个参数为 callback 的方法 b(CallBack callback)——背景 3
  4. A 的对象 a 在自己的 a() 里调用 B 的方法 b(CallBack callback)——A 类调用 B 类的某个方法
  5. 然后 b 就可以在 b(CallBack callback) 中调用 A 的方法——B 类调用 A 类的某个方法

综上:

  • 类 A 的 a() 调用类 B 的 b()。
  • 类 B 的 b() 执行完毕主动调用类 A 的 callback()。

该调用方式如图,也就是一种双向的调用方式。回调函数是一个函数或过程,不过它是一个由调用方自己实现,供被调用方使用的特殊函数。

二、理解

1️⃣老师在黑板上写一个式子 “8+8=”,由小黑来填空。小黑自己完成计算,模拟代码如下:

public class Student {
    private String name;
    public Student(String name) {
        this.name = name;
    }
    private int calcAdd(int a, int b) {
        return a + b;
    }
    public void fillBlank(int a, int b) {
        int result = calcAdd(a, b);
        System.out.println(name + "心算:" + a + "+" + b + "=" + result);
    }
}

小黑在填空(fillBlank)的时候,直接心算(clacAdd)了一下,得出结果是 2,并将结果写在空格里。测试如下:

public static void main(String[] args) {
    int a = 1;
    int b = 1;
    Student stu = new Student("小黑");
    stu.fillBlank(a, b);
}

该过程完全由 Student 类的实例对象单独完成,并未涉及回调机制。

2️⃣老师又在黑板上写了 “222+666=”,让小黑完成,然后回办公室去了。小黑懵逼的时候,小白同学递过来一个只能计算加法的计算机,小黑通过计算器计算得到结果并完成了填空。
计算器的代码为:

public class Calculator {
    public int add(int a, int b) {
        return a + b;
    }
}

修改 Student 类,添加使用计算器的方法:

public class Student {
    private String name;
    public Student(String name) {
        this.name = name;
    }
    private int useCalculator(int a, int b) {
        return new Calculator().add(a, b);
    }
    public void fillBlank(int a, int b) {
        int result = useCalculator(a, b);
        System.out.println(name + "使用计算器:" + a + "+" + b + "=" + result);
    }
}

测试如下:

public static void main(String[] args) {
    int a = 222;
    int b = 666;
    Student stu = new Student("小黑");
    stu.fillBlank(a, b);
}

该过程中仍未涉及到回调机制,但是小黑的部分工作已经实现了转移,由计算器来协助实现。

3️⃣老师又在黑板上写下了“1357+2468=”,让小黑上课之前完成填空,然后又回办公室了。小黑看着计算机,心里想着让小白代劳。小黑告诉小白题目,指出填写结果的具体位置,然后快乐的玩耍去了。
这里,将计算器和小白抽象成一个会算结果还会填空的超级计算器。该超级计算器需要传的参数是两个加数和要填空的位置,而这些内容需要小黑提前告知,也就是要把自己的一部分方法暴露出来,最简单的方法就是把自己的引用和两个加数一块告诉该超级计算器。
因此,超级计算器的 add 方法应该包含两个操作数和小黑自身的引用,代码如下:

public class SuperCalculator {
    public void add(int a, int b, Student stu) {
        int result = a + b;
        stu.fillBlank(a, b, result);
    }
}

小黑目前只需要一个方法“可以向超级计算器寻求帮助”就行了,代码如下:

public class Student {
    private String name;
    public Student(String name) {
        this.name = name;
    }
    public void callHelp(int a, int b) {
        new SuperCalculator().add(a, b, this);
    }
    public void fillBlank(int a, int b, int result) {
        System.out.println(name + "求助计算:" + a + "+" + b + "=" + result);
    }
}

测试如下:

public static void main(String[] args) {
    int a = 1357;
    int b = 2468;
    Student stu = new Student("小黑");
    stu.callHelp(a, b);
}

执行流程:
小黑通过自身的 callHelp() 调用了 new SuperCalculator() 的 add(),在调用的时候将自身的引用 this 当作参数一并传入,超级计算器在得出结果之后,回调了小黑的 fillBlank(),将结果填在了黑板的空格上。如此就体现了回调。此时,小黑的 fillBlank() 就是常说的回调函数。

通过这种方式,可以明显的看出,对于完成老师的填空题这个问题上,小黑已经不需要等待到加法做完且结果填写在黑板上才能去跟小伙伴撒欢了,填空这个工作由超级计算器小白来做了。回调的优势已经开始体现了。

4️⃣门卫大爷听到了小黑跟小伙伴们吹嘘自己如何在小白的帮助下答题。于是,决定找到超计来做自己的小帮手。

上述代码,超计的 add() 需要的参数是两个整型变量和一个 Student 对象,但是门卫不是学生,这里需要修改。如此,想到继承和多态。如果让小黑和门卫从一个父类进行继承,那么只需要给超计传入一个父类的引用就可以啦。

不过,Java 的单继承,以及不希望暴露太多,这里使用从接口继承的方式配合内部类来做。

换句话说,小白希望以后继续向班里的同学提供计算服务,同时还能向门卫提供算账服务,甚至以后能够拓展其他人的业务,于是向所有的顾客约定了一个办法,用于统一的处理,也就是自己需要的操作数和做完计算之后应该怎么做。这个统一的方法,小白做成了一个接口,提供给了大家,代码如下:

public interface doJob {
    void fillBlank(int a, int b, int result);
}

同时,小白修改了自己的计算器,使其可以同时处理不同的实现了 doJob 接口人的业务,代码如下:

public class SuperCalculator {
    public void add(int a, int b, doJob customer) {
        int result = a + b;
        customer.fillBlank(a, b, result);
    }
}

小黑和门卫拿到这个接口之后,只要实现了这个接口,就相当于按照统一的模式告诉小白得到结果之后的处理办法,按照之前说的使用内部类来做,代码如下:
小黑:

public class Student {
    private String name = null;
    public Student(String name) {
        this.name = name;
    }
    public class doHomeWork implements doJob {
        @Override
        public void fillBlank(int a, int b, int result) {
            System.out.println(name + "求助计算:" + a + "+" + b + "=" + result);
        }
    }
    public void callHelp(int a, int b) {
        new SuperCalculator().add(a, b, new doHomeWork());
    }
}

门卫:

public class Guard {
    private String name;
    public Guard(String name) {
        this.name = name;
    }
    public class doHomeWork implements doJob {
        @Override
        public void fillBlank(int a, int b, int result) {
            System.out.println(name + "求助算账:" + a + "+" + b + "=" + result + "元");
        }
    }
    public void callHelp(int a, int b) {
        new SuperCalculator().add(a, b, new doHomeWork());
    }
}

测试如下:

public static void main(String[] args) {
    int a = 66;
    int b = 88;
    int c = 1357;
    int d = 2468;
    Student stu = new Student("小黑");
    Guard guard = new Guard("门卫");
    stu.callHelp(a, b);
    guard.callHelp(c, d);
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,589评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,615评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,933评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,976评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,999评论 6 393
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,775评论 1 307
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,474评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,359评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,854评论 1 317
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,007评论 3 338
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,146评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,826评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,484评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,029评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,153评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,420评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,107评论 2 356

推荐阅读更多精彩内容