invokespecial和invokevirtual指令

1.JVM提供了5种方法调用指令,其作用列举如下:

  • invokestatic:该指令用于调用静态方法,即使用 static 关键字修饰的方法;
  • invokespecial:该指令用于三种场景:调用实例构造方法,调用私有方法(即private关键字修饰的方法)和父类方法(即super关键字调用的方法);
  • invokeinterface:该指令用于调用接口方法,在运行时再确定一个实现此接口的对象;
  • invokevirtual:该指令用于调用虚方法(就是除了上述三种情况之外的方法);
  • invokedynamic:在运行时动态解析出调用点限定符所引用的方法之后,调用该方法;在JDK1.7中推出,主要用于支持JVM上的动态脚本语言(如Groovy,Jython等)。

2.下面看一段代码例子

public class Test {
    static class Father {
        public void say() {
            System.out.println("i am fater");
            System.out.println(this);
            this.hello();
            this.hi();
        }

        private void hello() {
            System.out.println("father say hello");
        }

        public void hi() {
            System.out.println("father say hi");
        }
    }

    static class Son extends Father {

        public void hello() {
            System.out.println("son say hello");
        }

        public void hi() {
            System.out.println("son say hi");
        }
    }

    public static void main(String[] args) throws InterruptedException, FileNotFoundException {
        Father test = new Son();
        test.say();
    }
}

输出结果

i am fater
Test$Son@15db9742
father say hello
son say hi

先用javac Test.java 去编译源代码,生成
在这里插入图片描述
然后 用javap -v Test$Father.class ,查看jvm指令

在这里插入图片描述
我们重点关注say方法
第一行

0: getstatic     #2   // Field java/lang/System.out:Ljava/io/PrintStream;

getstatic指令: 获取类的静态字段,将其值压入栈顶。
在这里插入图片描述
也就是将System类中的静态变量out压栈

第二行

3: ldc           #3    // String i am fater

ldc指令 : int、float或String型常量从常量池推送至栈顶

第三行

5: invokevirtual #4   // Method java/io/PrintStream.println:(Ljava/lang/String;)V

调用PrintStream的println方法.

第四行
同第一行

第五行

11: aload_0

将this引用变量入栈

第六行

12: invokevirtual #5   // Method java/io/PrintStream.println:(Ljava/lang/Object;)V

打印this

第七,八行

15: aload_0
16: invokespecial #6     // Method hello:()V

这里是 invokespecial ,说明是执行的本实例的方法即Father类的hello方法.

第九,十行

19: aload_0
20: invokevirtual #7   // Method hi:()V

这里是invokevirtual 指令,说明是调用的虚方法,也就是执行的子类的方法。

上面就已经解释了,什么时候调子类的,什么时候是父类的,通过看jvm指令就知道了。

  • 6
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
字节码和机器码都是计算机可执行的指令集,但它们之间有很大的区别。 字节码是一种中间形式的代码,是Java源代码编译后生成的二进制文件,它并不是直接在计算机上运行的指令,而是由Java虚拟机(JVM)解释执行的。Java虚拟机是一个在操作系统上运行的应用程序,它能够将字节码翻译成机器码并执行。因此Java程序可以在任何支持Java虚拟机的平台上运行,这也是Java跨平台的主要原因之一。 机器码是计算机可以直接执行的指令集,它是由汇编语言编写的代码经过汇编器汇编后生成的二进制文件。机器码是一种特定的二进制编码,它是由计算机硬件所能直接识别和执行的。 举个例子,假设我们有一段Java代码: ```java public class HelloWorld { public static void main(String[] args) { System.out.println("Hello, World!"); } } ``` 当我们把这段代码编译成字节码后,可以使用`javap`命令来查看它的内容: ``` $ javac HelloWorld.java $ javap -c HelloWorld.class Compiled from "HelloWorld.java" public class HelloWorld { public HelloWorld(); Code: 0: aload_0 1: invokespecial #1 // Method java/lang/Object."<init>":()V 4: return public static void main(java.lang.String[]); Code: 0: getstatic #2 // Field java/lang/System.out:Ljava/io/PrintStream; 3: ldc #3 // String Hello, World! 5: invokevirtual #4 // Method java/io/PrintStream.println:(Ljava/lang/String;)V 8: return } ``` 可以看到,这段字节码并不是直接可执行的指令,而是类似汇编语言的一种中间形式。当我们运行这个程序时,Java虚拟机会将字节码解释成机器码并执行。 相比之下,如果我们使用汇编语言编写这个程序,会得到以下的汇编代码: ``` section .data db 'Hello, World!', 10, 0 section .text global _start _start: ; write message to stdout mov eax, 4 mov ebx, 1 mov ecx, message mov edx, 14 int 0x80 ; exit mov eax, 1 xor ebx, ebx int 0x80 message: db 'Hello, World!', 10, 0 ``` 这段代码是直接可执行的汇编代码,它是由汇编器汇编后生成的二进制文件。当我们运行这个程序时,计算机可以直接执行这段机器码。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值