如何用匿名内部类实现 Java 同步回调

在一个应用系统中,不论使用何种编程语言,模块之间要进行调用,仅存在三种方式:同步调用、异步调用、回调。本文就其中回调方式进行详细解读,并通过匿名内部类的手段,在最后实现一个同步回调的过程。

一、回调的意义

在学习回调之前,我们需要知道使用回调的原因,和回调的应用场景。

不如先思考两个问题:

  • 栈底对栈顶通常是不可见的,但是栈顶有时需要直接调用栈底
  • 上级派下级做事,在此期间,下级可能需要通过上级获取高权限的协助

而在本例中,回调方式被用来处理爬取后的大量返回数据。在业务层面,这些数据被安排在调用方进行处理,但是调用方却没有处理这些数据的足够权限。于是,通过回调,业务被很好的分层并且执行。

二、如何实现同步回调

本文对同步回调的业务需求如下:

  1. 回调方调用调用方进行数据爬取
  2. 调用方调用回调方进行数据存储
  3. 调用方调用回调方进行日志记录

根据需求可以得到回调过程的时序图:

相应代码如下:

public interface Handler {
  void handle(String info);
}

public class Task {
  private String info;

  private void setInfo(String info) {
    this.info = info;
  }

  public void call() {
    Crawler.getInstance().crawl(new Handler() {
      @Override
      public void handle(String info) {
        setInfo(info);
      }
    });
  }
}

public class Crawler {
  private static Crawler instance = null;

  public static Crawler getInstance() {
    if (instance == null) {
      instance = new Crawler();
    }
    return instance;
  }

  private String getInfo() {
    return "the info from crawler";
  }

  public void crawl(Handler handler) {
    handler.handle(getInfo());
  }
}

三、遇到的问题

如果我们使用代码来实现上述回调过程,不难会发现这样一个问题:Task调用Crawler,Crawler调用Handler,Hanlder调用Task。很明显,此处存在一个环,产生了循环依赖的问题,而接口可以为我们提供良好的解决方案。

四、为什么通过匿名内部类的方式

用 Java 实现同步回调有许多方式,为什么我们要通过匿名内部类的方式来实现回调,直接回调不香吗?

不妨先看看直接回调的顺序图:

相应代码如下:

public interface Handler {
  void handle(String info);
}

public class Task implements Handler{
  private String info;

  private void setInfo(String info) {
    this.info = info;
  }

  public void call() {
    Crawler.getInstance().crawl(this);
  }

  @Override
  public void handle(String info) {
    setInfo(info);
  }
}

public class Crawler {
  private static Crawler instance = null;

  public static Crawler getInstance() {
    if (instance == null) {
      instance = new Crawler();
    }
    return instance;
  }

  private String getInfo() {
    return "the info from crawler";
  }

  public void crawl(Handler handler) {
    handler.handle(getInfo());
  }
}

直接回调带来的最大问题便是回调接口的暴露,也就是说回调接口不一定用于回调,也可以用于直接访问。这在业务层面的设计上是绝对不允许的,而匿名内部类在执行回调等特定业务的同时,可以很好的对外隐藏用于回调的接口。

五、总结

常规类通常无法对回调等特定接口作出限定,要么都可以访问,要么都拒绝访问。而内部类通过牺牲自身的被访问权限,提升了自身访问外部类的能力,这使得其成为实现回调的首选方案。在JAVA8中,lambda表达式本质上就是匿名内部类的语法糖。

注:匿名内部类本质上是成员内部类、局部内部类的简化写法,这里将其统称为内部类。

参考链接

[1] <<Java核心技术>> 卷一

[2] https://www.cnblogs.com/xrq730/p/6424471.html

以上就是如何用匿名内部类实现 Java 同步回调的详细内容,更多关于Java 同步回调的资料请关注我们其它相关文章!

(0)

相关推荐

  • Java匿名对象与匿名内部类

    匿名对象:没有名字的对象. 非匿名对象: ClassName c=new ClassName(); c.run(); 匿名对象: new ClassName().run(); 注意事项: 1.当对象对方法仅进行一次调用的时候,就可以简化成匿名对象. 2.两个匿名对象不可能是同一个对象. 3.一般不给匿名对象赋予属性值,因为永远无法获取到. 4.运行一次,直接就被回收掉了,节省内存空间. 匿名对象使用的代码例子: public class Anony{ int a=1; int b=2; void

  • Java Lambda表达式与匿名内部类的联系和区别实例分析

    本文实例讲述了Java Lambda表达式与匿名内部类的联系和区别.分享给大家供大家参考,具体如下: 一 点睛 Lambda表达式与匿名内部类存在如下相同点: Lambda表达式与匿名内部类一样,都可以直接访问"effectively final"的局部变量,以及外部类的成员变量(包括实例变量和类变量). Lambda表达式创建的对象与匿名内部类生成的对象一样, 都可以直接调用从接口继承得到的默认方法. Lambda表达式与匿名内部类主要存在如下区别: 匿名内部类可以为任意接口创建实例

  • Java中的接口回调实例

    定义: /** * @author Administrator * @project: TestOne * @package: PACKAGE_NAME * @date: 2018/11/30 0030 15:42 * @brief: 郭宝 **/ public class Person { /** * 自定义一个接口 **/ public interface OnNameChangeListener{ //接口中的抽象函数,并携带数据 void onNameChange(String name

  • 详解APP微信支付(java后台_统一下单和回调)

    1.微信配置信息 global.properties 2.方法wxpay用于生成预支付订单信息 方法notifyWeiXinPay用于微信支付成功后的回调, 注意: 在手机端使用微信支付成功后,微信服务器会根据提供的回调地址进行回调, parameterMap.put("notify_url", wxnotify); (见下面代码) 在局域网是无法进行回调的,必须将你的服务端放在公网上进行测试, 回调函数会被多次调用,如果第一次成功后,你可以将业务数据状态标志为已处理, 对于相同订单的

  • Java匿名类,匿名内部类实例分析

    本文实例讲述了Java匿名类,匿名内部类.分享给大家供大家参考,具体如下: 本文内容: 内部类 匿名类 首发日期 :2018-03-25 内部类: 在一个类中定义另一个类,这样定义的类称为内部类.[包含内部类的类可以称为内部类的外部类] 如果想要通过一个类来使用另一个类,可以定义为内部类.[比如苹果手机类,苹果手机类中的黄金版的是特别定制的] 内部类的外部类的成员变量在内部类中仍然有效,内部类中的方法也可以调用外部类中的方法.[不论是静态还是非静态的,内部类都可以直接调用外部类中的属性,] 内部

  • 详解java 三种调用机制(同步、回调、异步)

    1:同步调用:一种阻塞式调用,调用方要等待对方执行完毕才返回,它是一种单向调用 2:回调:一种双向调用模式,也就是说,被调用方在接口被调用时也会调用对方的接口: 3:异步调用:一种类似消息或事件的机制,不过它的调用方向刚好相反,接口的服务在收到某种讯息或发生某种事件时,会主动通知客户方(即调用客户方的接口 具体说来:就是A类中调用B类中的某个方法C,然后B类中反过来调用A类中的方法D,D这个方法就叫回调方法, 实例1:使用java中Timer来在给定时间间隔发送通知,每隔十秒打印一次数据 Tim

  • Java线程的start方法回调run方法的操作技巧

    面试中可能会被问到为什么我们调用start()方法时会执行run()方法,为什么我们不能直接调用run()方法? Java 创建线程的方法 实际上,创建线程最重要的是提供线程函数(回调函数),该函数作为新创建线程的入口函数,实现自己想要的功能.Java 提供了两种方法来创建一个线程: 继承 Thread 类 class MyThread extends Thread{ public void run() { System.out.println("My thread is started.&qu

  • Java匿名内部类的写法示例

    前言 在Java中调用某个方法时,如果该方法的参数是一个接口类型,除了可以传入一个参数接口实现类,还可以使用匿名内部类实现接口来作为该方法的参数. 匿名内部类其实就是没有名称的内部类,在调用包含有接口类型参数的方法时,通常为零简化代码,不会创建一个接口的实现类作为方法参数传入,而是直接通过匿名内部类的形式传入一个接口类型参数,在匿名内部类中直接完成方法的实现. 创建匿名内部类的基本语法格式如下: new 父接口(){     //匿名内部类实现部分 } 示例 interface Animal{

  • java-RGB调色面板的实现(事件监听器之匿名内部类)

    题目要求: 要求写一个案例,使用三个JSlider分别选取R.G.B三原色的值,用户可以通过活动JSlider的滑块来动态的合成一种颜色,合成的颜色显示在界面上. 代码思路解析: 1.所需要的组件 (1)JFrame窗口  (2)JLabel的RGB标签文字组件  (3)JTextField颜色变动的数值  (4)JSlider颜色交换的滑块组件  (5)JTextArea颜色显示区域 2.构造方法中将JFrame.JSlider.JTextField.JTextArea组件全部实例化 3.另写

  • Java为什么匿名内部类参数引用需要用final进行修饰?

    事实上,除了匿名内部类参数,方法和作用域内的内部类内部使用的外部变量也必须是final 的.原因大致总结一下: 简单解释就是: 方法中的局部变量的生命周期很短,方法结束后变量就要被销毁,加上final是为了延长变量的生命周期. 进一步解释: 内部类通常都含有回调,引用那个匿名内部类的函数执行完了就没了,所以内部类中引用外面的局部变量需要是final的,这样在回调的时候才能找到那个变量,而如果是外围类的成员变量就不需要是final的,因为内部类本身都会含有一个外围了的引用(外围类.this),所以

随机推荐