Java接口回调的本质详解

本质是JVM指令invokevirtual它涉及到了多态的特性,使用 virtual dispatch 做方法调用

virtual dispatch 机制会首先从 receiver(被调用方法的对象的实际类型)的类的实现中查找对应的方法,如果没找到,则去父类查找,直到找到函数并实现调用,而不是依赖于引用的类型。

我们知道了接口回调的本质是多态,多态的本质是JVM的invokevirtual指令,那么我们就不用拘泥于接口和抽象类或者继承关系这种概念了,我们直接写一个用普通类实现的回调然后再分析它:

public class TestA{

    public void A(){
        System.out.println("A");
    }

}
public class TestC {

    public void C(TestA A){
        System.out.println("ccc1");
        A.A();
        System.out.println("ccc2");
    }

}
public class JavaTest {
    public static void main(String[] args) {

        TestC testC = new TestC();

        testC.C(new TestA(){
            @Override
            public void A() {
                System.out.println("fsf");
            }
        });
    }
}

执行结果:

如果是invokespecial指令:

public class TestA {
    private void A(){
        System.out.println("A");
    }
}

public class TestC {
    public void C(TestA A){
        System.out.println("ccc1");
//注意这里不能用 A.getClass,getClass方法也是用invokevirtual指令调用的
        Class aClass = TestA.class;
        try {
          Method method = aClass.getDeclaredMethod("A");
          method.setAccessible(true);
          method.invoke(A);
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println("ccc2");
    }
}

public class Test{

    public static void main(String[] args) {
        TestC testC = new TestC();
        testC.C(new TestA(){
            private void A() {
                System.out.println("fsf");
            }
        });
    }

}

JVM在调用private方法的时候使用的是invokespecial指令,但是我们不能直接在外部调用私有方法所以就写了个反射来调用,执行结果:

如果有因为是不是因为反射导致的,我们排除一下反射:

public class TestA{
    public void A() {
        System.out.println("A");
    }
}

public class TestC {
    public void C(TestA A){
        System.out.println("ccc1");
        Class aClass = TestA.class;
        try {
          Method method = aClass.getDeclaredMethod("A");
          method.setAccessible(true);
          method.invoke(A);
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println("ccc2");
    }
}

public class Test{

    public static void main(String[] args) {
        TestC testC = new TestC();
        testC.C(new TestA(){
            public void A() {
                System.out.println("fsf");
            }
        });
    }

}

执行结果:

使用反射和不使用反射的结果一致可以排除反射的干扰

到此这篇关于Java接口回调的本质详解的文章就介绍到这了,更多相关Java接口回调内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • 深入了解Java接口回调机制

    一.前言 最近在看android fragment与Activity进行数据传递的部分,看到了接口回调的内容,今天来总结一下. 二.回调的含义和用途 1.什么是回调? 一般来说,模块之间都存在一定的调用关系,从调用方式上看,可以分为三类同步调用.异步调用和回调.同步调用是一种阻塞式调用,即在函数A的函数体里通过书写函数B的函数名来调用之,使内存中对应函数B的代码得以执行.异步调用是一种类似消息或事件的机制解决了同步阻塞的问题,例如 A通知 B后,他们各走各的路,互不影响,不用像同步调用那样, A

  • 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

  • Java按时间梯度实现异步回调接口的方法

    1. 背景 在业务处理完之后,需要调用其他系统的接口,将相应的处理结果通知给对方,若是同步请求,假如调用的系统出现异常或是宕机等事件,会导致自身业务受到影响,事务会一直阻塞,数据库连接不够用等异常现象,可以通过异步回调来防止阻塞,但异步的情况还存在一个问题,若调用一次不成功的话接下来怎么处理?这个地方就需要按时间梯度回调,比如前期按10s间隔回调,回调3次,若不成功按30s回调,回调2次,再不成功按分钟回调,依次类推--相当于给了对方系统恢复的时间,不可能一直处于异常或宕机等异常状态,若是再不成

  • 基于Java接口回调详解

    目录 Java接口回调 产生接口回调的场景 什么是接口回调 在代码中的使用 java接口回调测试 接口类 实现类 回调 实现测试 Java接口回调 产生接口回调的场景 在现实生活中,产生接口回调的场景很简单,比如我主动叫你帮我做一件事,然后你做完这件事之后会通知我,"通知我"这个动作就是接口回调的动作.接口回调重在理解思就好.再举个例子用于下面的代码演示,老师让学生做课堂练习,学生做完练习后并告诉老师. 什么是接口回调 在Java的层面简单的说一下接口回调:现有一个接口A,和它的实现类

  • java 接口回调实例详解

    java 接口回调实例详解 首先官方对接口回调的定义是这样的,所谓回调:就是A类中调用B类中的某个方法C,然后B类中反过来调用A类中的方法D,D这个方法就叫回调方法.这样听起来有点绕,我们可以这么理解接口回调:比如我们想知道隔壁老王啥时候回家?但是我们有自己的事情做不能一直监视着老王,那么我们可以雇员小区的保安来完成这个任务,当老王回家口,保安就给我们打电话告诉我们,老王回来了!这样就完成了一个事件的传递: 首先我们定义了一个接口: public interface DynamicMessage

  • Java接口回调的本质详解

    本质是JVM指令invokevirtual它涉及到了多态的特性,使用 virtual dispatch 做方法调用 virtual dispatch 机制会首先从 receiver(被调用方法的对象的实际类型)的类的实现中查找对应的方法,如果没找到,则去父类查找,直到找到函数并实现调用,而不是依赖于引用的类型. 我们知道了接口回调的本质是多态,多态的本质是JVM的invokevirtual指令,那么我们就不用拘泥于接口和抽象类或者继承关系这种概念了,我们直接写一个用普通类实现的回调然后再分析它:

  • Java接口和抽象类原理详解

    这篇文章主要介绍了Java接口和抽象类原理详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 对于面向对象编程来说,抽象是它的一大特征之一.在Java中,可以通过两种形式来体现OOP的抽象:接口和抽象类.这两者有太多相似的地方,又有太多不同的地方.很多人在初学的时候会以为它们可以随意互换使用,但是实际则不然.今天我们就一起来学习一下Java中的接口和抽象类.下面是本文的目录大纲: 一.抽象类 在了解抽象类之前,先来了解一下抽象方法.抽象方法是一

  • Java的回调机制实例详解

    本文实例讲述了Java的回调机制.分享给大家供大家参考,具体如下: 一 代码 interface CallBack // 回调接口 { void methodToCallBack( ); //回调方法 } class CallBackImpl implements CallBack // 实现回调接口和回调方法 { public void methodToCallBack( ) { System.out.println("I've been called back"); } } pub

  • java回调机制实例详解

    java回调机制实例详解 以前不理解什么叫回调,天天听人家说加一个回调方法啥的,心里想我草,什么叫回调方法啊?然后自己就在网上找啊找啊找,找了很多也不是很明白,现在知道了,所谓回调:就是A类中调用B类中的某个方法C,然后B类中反过来调用A类中的方法D,D这个方法就叫回调方法,这样子说你是不是有点晕晕的,其实我刚开始也是这样不理解,看了人家说比较经典的回调方式: Class A实现接口CallBack callback--背景1 class A中包含一个class B的引用b --背景2 clas

  • Java Collection 接口和常用方法综合详解

    目录 1. Collection 接口实现类的特点 2. Collection 接口常用方法 3. Collection接口遍历元素方式1- 使用Iterator(迭代器) 4. Collection 接口遍历对象方式2- for 循环增强 5. 练习 1. Collection 接口实现类的特点 public interface Collection<E> extends Iterable<E> 1)Collection实现子类可以存放多个元素,每个元素可以是Object 2)有

  • Java 回调callback举例详解

    目录 前言 1.什么是回调 2.例子 第一章.故事的缘起 第二章.幼师的找茬 第三章.幼师回来了 第四章.门口的婆婆 前言 回调的核心就是回调方将本身即this传递给调用方,这样调用方就可以在调用完毕之后告诉回调方它想要知道的信息. 1.什么是回调 软件模块之间总是存在一定的接口,从调用方式上,可以把他们分为三类:同步调用.回调和异步调用. (1)同步调用: 同步调用是最基本并且最简单的一种调用方式,类A的方法a()调用类B的方法b(),一直等待b()方法执行完毕,a()方法再继续往下走.这种调

  • Java集合之Map接口与实现类详解

    目录 初识Map Map中常用方法 HashMap LinkedHashMap TreeMap HashMap和TreeMap的比较 Hashtable 集合中元素的遍历 iterator接口中的方法 Enumeration接口中的方法 初识Map Map接口没有从Collection接口继承,Map接口用于维护“键-值”对数据,这个“键-值”对就是Map中的元素,Map提供“键(Key)”到“值(value)”的映射,一个Map中键值必须是唯一的,不能有重复的键,因为Map中的“键-值”对元素

  • java反射之通过反射了解集合泛型的本质(详解)

    本文接上文"java反射之方法反射的基本操作方法",利用反射了解下java集合中泛型的本质 1.初始化两个集合,一个使用泛型,一个不使用 ArrayList list1 = new ArrayList(); ArrayList<String> list2 = new ArrayList<String>(); 2.有定义类型可得在list2中添加int类型会报错 list2.add("Hello"); list2.add(20); //报错 3

随机推荐