全面掌握同步、回调、异步回调的使用

本文主要解决如何理解同步调用、回调、异步调用的概念及其相互关系,然后从实例的角度来阐述它们的使用方法。

我们知道,软件众多模块之间如果要进行通信,都必须通过相互的接口进行调用,那么如何调用就成了一个问题,现如今主要存在以下三种方式的调用:

  • 同步阻塞调用
  • 回调
  • 异步回调

一、同步调用

关于同步调用有以下几个基本认知:

  • 它是一种阻塞型的调用方式,调用方总是要等到被调用方执行结束或者返回结果后才能继续往下执行;
  • 它是一种单向的调用,只存在调用方调用被调用方的行为。

同步调用很好理解,下面是一个最基本的示例:

public class Student {
    public void doHomework(){
        System.out.println("student——homework");
    }
}
public class Teacher {  
    public void giveLecture(){
        System.out.println("teacher——lecture");
    }   
    public void assignHomework(Student rose){
        System.out.println("teacher——assign——homework");
        // 同步调用
        rose.doHomework();
    }   
    public void fixHomework(){
        System.out.println("teacher——fix——homework");
    }
}
public class Test {
    public static void main(String[] args) {
        Teacher jack = new Teacher();
        Student rose = new Student();
        jack.giveLecture();
        jack.assignHomework(rose);
        jack.fixHomework();
    }
}

在上述代码中,teacher主动调用了studentdoHomework方法,只有等到该方法执行完成以后,teacher的其它行为才会执行,代码的执行结果如下:

teacher——lecture
teacher——assign——homework
student——homework
teacher——fix——homework

二、回调

关于同步调用有以下几个基本认知:

  • 回调也是一种阻塞型的调用方式,调用方也总是要等到被调用方执行结束或者返回结果后才能继续往下执行;
  • 回调是一种双向调用的模式,被调用方也可以调用调用方。

基于上面的例子,我们来通过代码看一下回调的实现逻辑:

public class Student {
    public void doHomework(Teacher teacher){
        System.out.println("student——homework");
        // 回调teacher中的方法
        teacher.fixHomework();
    }
}
public class Teacher {
    public void giveLecture(){
        System.out.println("teacher——lecture");
    }
    public void assignHomework(Student rose){
        System.out.println("teacher——assign——homework");
        // 调用student的方法时,将自身的引用也传递过去,以实现回调
        rose.doHomework(this);
    }
    public void fixHomework(){
        System.out.println("teacher——fix——homework");
    }
    public void sayGoodbye(){
        System.out.println("teacher——goodbye!");
    }
}
public class Test {
    public static void main(String[] args) {
        Teacher jack = new Teacher();
        Student rose = new Student();
        jack.giveLecture();
        jack.assignHomework(rose);
        jack.sayGoodbye();
    }
}

在上面的例子中,teacher在调用student的方法时,将自身的引用对象也传递过去了,这样student中的该方法就可以在自己认为合适的时候回调teacher中的某个方法,这就是回调,代码的执行结果如下:

teacher——lecture
teacher——assign——homework
student——homework
teacher——fix——homework
teacher——goodbye!

在本例中,使用回调的好处非常明显,teacherfixHomework不需要在main中主动调用,当studentdoHomework结束的时候自动就可以执行了。

三、异步回调

在上面回调的例子中,我们可以看到,它其实还是同步阻塞式的,也就意味着,studentdoHomework没有结束,teacher的后续行为都无法进行,这在异步调用中就可以解决这一问题。

我们还是使用上面的例子,只不过异步的实现需要借助Java的Thread来实现,具体的示例如下:

public class Student {
    public void doHomework(Teacher teacher){
        System.out.println("student——homework");
        // 回调teacher中的方法
        teacher.fixHomework();
    }
}
public class Teacher {
    public void giveLecture(){
        System.out.println("teacher——lecture");
    }
    // 备注一:这里为什么要使用final?
    public void assignHomework(final Student rose){
        System.out.println("teacher——assign——homework");
        // 启动一个线程来实现异步调用
        new Thread(
            new Runnable(){
                @Override
                public void run() {
                    // 备注二:这里为什么要使用Teacher.this而不是this?
                    rose.doHomework(Teacher.this);
                }   
            }
                ).start();
    }
    public void fixHomework(){
        System.out.println("teacher——fix——homework");
    }
    public void watchTv(){
        System.out.println("teacher——watchTv");
    }
}
public class Test {
    public static void main(String[] args) {
        Teacher jack = new Teacher();
        Student rose = new Student();
        jack.giveLecture();
        jack.assignHomework(rose);
        jack.watchTv();
    }
}

在这个例子中,teacher在assignHomework的时候启动了线程来实现异步地调用student的doHomework,然后不管doHomework是否完成都继续往下执行,然后student的doHomework完成以后才回调,这就实现了异步回调,代码的执行结果如下:

teacher——lecture
teacher——assign——homework
teacher——watchTv
student——homework
teacher——fix——homework

最后要需要对代码中的备注进行说明:

  • 备注一:其实上面例子中用到的线程实例其实是一个匿名内部类,在该匿名内部类中使用外部类方法中的形参需要把该形参定义为final类型;
  • 备注二:同样是基于匿名内部类的原因,此处使用this指代的将是当前的线程对象,而不是外部类teacher的实例引用对象,所以我们需要使用Teacher.this来强制使用外部类的实例引用对象传递给student,从而实现回调。

关于内部类的详细说明和用法,在另一篇文章《内部类有哪些?它们存在的意义是什么?》中有非常详细的介绍。

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

推荐阅读更多精彩内容