22.5种方法调用指令-invokestatic,invokespecial,invokevirtual,invokeinterface,invokedynamic

1.虚方法与非虚方法

1.静态方法、私有方法、final方法、实例构造器、父类方法都是非虚方法,其他方法都是虚方法。
在这里插入图片描述

2. 5种调用指令

a)invokestatic
b)invokespecial
c)invokevirtual
d)invokeinterface
e)invokedynamic
invokestaticinvokespecial调用的方法都是非虚方法,其余的方法(final修饰的方法除外,final方法调用使用的是invokevirtual,但是final方法不能重写,所以认为final是非虚方法)都是虚方法。
在这里插入图片描述
例子:

package jvn;

class Father {
    public Father() {
        System.out.println("father的构造器");
    }

    public static void showStatic(String str) {
        System.out.println("father " + str);
    }

    public final void showFinal() {
        System.out.println("father show final");
    }

    public void showCommon() {
        System.out.println("father 普通方法");
    }
}

public class Son extends Father {
    public Son() {
        //invokespecial
        super();
    }
    public Son(int age) {
        //invokespecial
        this();
    }
    //不是重写的父类的静态方法,因为静态方法不能被重写!
    public static void showStatic(String str) {
        System.out.println("son " + str);
    }
    private void showPrivate(String str) {
        System.out.println("son private" + str);
    }

    public void show() {
        //invokestatic
        showStatic("atguigu.com");
        //invokestatic
        super.showStatic("good!");
        //invokespecial
        showPrivate("hello!");
        //invokespecial
        super.showCommon();

        //invokevirtual
        showFinal();//因为此方法声明有final,不能被子类重写,所以也认为此方法是非虚方法。
        //虚方法如下:
        //invokevirtual
        showCommon();
        info();

        MethodInterface in = null;
        //invokeinterface
        in.methodA();
    }

    public void info(){

    }

    public void display(Father f){
        f.showCommon();
    }

    public static void main(String[] args) {
        Son so = new Son();
        so.show();
    }
}

interface MethodInterface{
    void methodA();
}

使用Jclasslib打开编译生成的class文件。选择show方法,查看字节码指令。
静态方法都是使用invokestatic调用。
私有方法是使用invokespecial调用。
父类的普通方法是通过invokespecial调用。
父类的final方法是通过invokevirtual调用。
接口方法使用invokeinterface调用。
在这里插入图片描述
e)关于invokedynamic指令
1.静态类型语言和动态类型语言的区别:对类型的检查是在编译期还是在运行期。
例如:
Java定义一个字符串变量:
String info = "test"; //如果写成info = test就会直接报错,根本不用运行。
而JS定义一个变量:
var name = “test”; // 只有等到js运行的时候才能确定这是一个字符串
在这里插入图片描述
1.java是静态类型语言。
2.java8为了支持动态类型语言开始支持invokedynamic
在这里插入图片描述
Java8中的lambda表达式会生成invokedynamic指令:

package jvn;
@FunctionalInterface
interface Func {
    public boolean func(String str);
}
public class Lambda {
    public void lambda(Func func) {
        return;
    }
    public static void main(String[] args) {
        Lambda lambda = new Lambda();
        Func func = s -> {
            return true;
        };
        lambda.lambda(func);
        lambda.lambda(s -> {
            return true;
        });
    }
}

使用Jclasslib打开上述代码生成的class文件:
在这里插入图片描述
更多JVM文章请访问我的JVM专栏:
https://blog.csdn.net/u011069294/category_10113093.html

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章