JVM学习-字节码指令集(二)

对象的创建与访问指令
创建指令
  • 虽然类实例和数组都是对象,但Java虚拟机对类实例和数组的创建和操作使用了不同的字节码指令
  • 创建类实例指令:new
  • 它接收一个操作数,指向常量池的索引,表示要创建的类型,执行完成后,将对象的引用压入操作数栈
  • 创建数组的指令
  • 创建数组的指令:newarray,arewarray,multianewarray
  • newarray:创建基本类型数组
  • anewarray:创建引用类型数组
  • multianewarray:创建多维数组
字段访问指令
  • 对象创建后,可能通过对象访问指令获得对象实例或数组实例中的字段或数组元素
  • 访问字段(static字段,或称为类变量)的指令:getstatic,putstatic
  • 访问类实例字段(非static字段,可实例变量):getfield,putfield
    如:以getstatic指令为例,它含有一个操作数,为指向常量池的Fieldref索引,它的作用是获取Fieldref指定的对象或者值,并将其压入操作数栈
public void sayHello() {
        System.out.println("hello");
    }
//对应字节码
0 getstatic #8 <java/lang/System.out>
3 ldc #9 <hello>
5 invokevirtual #10 <java/io/PrintStream.println>
8 return
数组操作指令
  • 数组操作指令主要有:xastore和xaload指令
  • 把一个数组元素加载到操作数栈的命令:baload,caload,saload,iaload,laload,faload,daload,aaload
  • 将一个操作数栈的值存储到数组元素中的指令:bastore,castore,sastore,iastore,lastore,fastore,dastore,aastore
    在这里插入图片描述
  • 取数组长度的指令:arraylength
  • 该指令弹出栈顶的数组元素,获取数组的长度,将长度压入栈
public void arrLength() {
        double[] arr = new double[10];
        System.out.println(arr.length);
    }
    //字节码
 0 bipush 10
 2 newarray 7 (double)
 4 astore_1
 5 getstatic #8 <java/lang/System.out>
 8 aload_1
 9 arraylength       //获取数组长度
10 invokevirtual #14 <java/io/PrintStream.println>
13 return
  • 说明
  • 指令xload表示将数组的元素压栈,如saload,caload表示压入short数组和char数组,指令xaload在执行时,要求操作数中栈顶元素为数组索引i,栈顶顺位第二个元素为数组引用a,该指令会弹出栈顶这两个元素,并将a[i]重新压入堆栈
  • xastore则专门针对数组操作,以iastore为例,它用于给一个int数组的给定索引赋值,在iastore执行前,操作数栈顶需要以此准备3个元素:值、索引、数组引用,istore会弹出这三个值,并将值败给数组中指定索引的位置
public void setArray() {
        int[] intArray = new int[10];
        intArray[3] = 20;
        System.out.println(intArray[1]);
    }
//字节码
 0 bipush 10
 2 newarray 10 (int)
 4 astore_1
 //----------------对应intArray[3] = 20
 5 aload_1           //数组地址
 6 iconst_3          //索引
 7 bipush 20         //值
 9 iastore
//----------------对应intArray[3] = 20
10 getstatic #8 <java/lang/System.out>
//----------------对应intArray[1]
13 aload_1        //数组地址
14 iconst_1       //数组索引
15 iaload
//----------------对应intArray[1]
16 invokevirtual #14 <java/io/PrintStream.println>
19 return
类型检查指令
  • 检查类实例或数组类型的指令:instanceof,checkcast
  • 指令checkcast用于检查类型强制转换是否可以进行,如果可以进行,那么checkcast指令不会改变操作数栈,否则它会抛出CassCastException异常
  • 指令instanceof用来判断是否是某一个类的实例,它会将判断结果压入操作数栈
public String checkCast(Object obj) {
        if (obj instanceof String) {
            return (String)obj;
        } else {
            return null;
        }
    }
//字节码指令
 0 aload_1
 1 instanceof #17 <java/lang/String>   //判断是否为String,即obj instanceof String
 4 ifeq 12 (+8)
 7 aload_1
 8 checkcast #17 <java/lang/String>       //强转
11 areturn
12 aconst_null
13 areturn
方法调用与返回指令
方法调用指令
  • 方法调用指令:invokevirtual,invokeinterface,invokespecial,invokestatic,invokedynamic
  • invokevirtual:用于调用对象的实例方法,根据对象的实际类型进行分派(虚方法分派),支持多态,是最常见的方法分派方式
  • invokeinterface:用于调用接口方法,它会在运行时搜索由特定对象所实现的这个接口方法,并找出适合的方法进行调用
//方法调用指令:invokeinterface
    public void invoke3() {
        Thread t1 = new Thread();
        ((Runnable)t1).run();
        Comparable<Integer> com = null;
        com.compareTo(123);
    }
  
//字节码
 0 new #4 <java/lang/Thread>
 3 dup
 4 invokespecial #5 <java/lang/Thread.<init>>
 7 astore_1
 8 aload_1
 9 invokeinterface #9 <java/lang/Runnable.run> count 1             //调用接口方法
14 aconst_null
15 astore_2
16 aload_2
17 bipush 123
19 invokestatic #10 <java/lang/Integer.valueOf>
22 invokeinterface #11 <java/lang/Comparable.compareTo> count 2
27 pop
28 return

在这里插入图片描述

  • invokespecial:调用一些特殊处理的实例方法,包含实例初始化方法(构造器),私有方法和父类方法,这些方法是静态类型绑定的,不会在调用时动态派发
//方法调用指令:invokespecial:静态分派
    public void invoke1() {
        //情况1:类实例构造器方法
        Date date = new Date();
        Thread t1 = new Thread();
        //情况2:调用父类方法
        super.toString();
        //情况3:私有方法
        methodPrivate();
    }

    private void methodPrivate() {}
//字节码指令
 0 new #2 <java/util/Date>
 3 dup
 4 invokespecial #3 <java/util/Date.<init>>
 7 astore_1
 8 new #4 <java/lang/Thread>
11 dup
12 invokespecial #5 <java/lang/Thread.<init>>        //构造器调用
15 astore_2
16 aload_0
17 invokespecial #6 <java/lang/Object.toString>        //父类方法调用
20 pop
21 aload_0
22 invokespecial #7 <com/chapter10/MethodInvokeReturnTest.methodPrivate>  //私有方法调用
25 return
  • invokestatic:调用命名类中的类方法(static方法),这是静态绑定的
//调用指令:invokestatic
    public void invoke2() {
        methodStatic();
    }

    public static void methodStatic() {}
//字节码
0 invokestatic #8 <com/chapter10/MethodInvokeReturnTest.methodStatic>
3 return
  • invokedynamic:调用动态绑定的方法,JDK1.7新加入的指令,用于在运行时动态解析出调用点限定符所引用的方法,并执行该方法,前4条调用指令的分派逻辑都固化在java虚拟机内部,而invokedynamic指令的分派逻辑是由用户所设定的引导方法决定的
方法返回指令
  • 方法调用结束前,需要进行返回,方法返回指令是根据返回值的类型区分的
  • 包括ireturn(boolan,byte,char,short和int类型使用),lreturn,freturn,dreturn和areturn
  • 还有一条return指令供声明为void的方法,实例初始化方法以及类和接口的类初始化方法使用
    在这里插入图片描述
  • 举例
  • 通过ireturn指令,将当前函数操作数栈的顶层元素弹出,并将这个元素压入调用者函数的操作数栈中,所有在当前函数操作数栈中的其他元素都会被丢弃
  • 如果当返回值是synchronized方法,那么还会执行一个隐含的monitorexit指令,退出临界区
  • 最后,会丢弃当前方法的整个帧,恢复调用者的帧,并将控制权转交给调用者
//方法返回指令
    public int returnInt() {
        int i = 200;
        return i;
    }
//字节码
0 sipush 200
3 istore_1
4 iload_1
5 ireturn
    public double returnDouble() {
        return 0.0;
    }
 //字节码
 0 dconst_0
1 dreturn
    public String returnString() {
        return "hello,world";
    }
//字节码
0 ldc #16 <hello,world>
2 areturn
    public int[] returnArr() {
        return null;
    }
        //字节码
    public float returnFloat() {
        int i = 10;
        return i;
    }
//字节码
0 bipush 10
2 istore_1
3 iload_1
4 i2f
5 freturn
    public byte returnByte() {
        return 0;
    }
//字节码
0 iconst_0
1 ireturn

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/660008.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

基于匹配追踪和最大重叠离散小波变换的ECG心电信号R波检测(MATLAB 2018a)

准确识别心电信号的R波是进行HRV分析的前提。因此&#xff0c;开发出准确的心电信号R波检测方法十分重要。近几十年来&#xff0c;提出的R峰检测方法主要分为两个阶段。第1阶段是预处理阶段&#xff0c;目的是对受不同噪声影响的原始心电信号进行降噪处理&#xff0c;从而实现增…

1794 jsp蛋糕店管理系统Myeclipse开发mysql数据库web结构java编程计算机网页项目

一、源码特点 jsp 蛋糕店管理系统 是一套完善的web设计系统&#xff0c;对理解JSP java编程开发语言有帮助采用了java设计&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统采用web模式&#xff0c;系统主要采用B/S模式开发。开发环境为TOMCAT7.0,Myeclipse8.5开发&…

Pycharm打开django支持

在 PyCharm 中打开 “Settings/Preferences” -> “Languages & Frameworks” -> “Django”。 勾上Enable Django support 然后配置好文件根目录就好了

STM32FLASH闪存

文章目录 前言首先来回顾一下存储器映像FLASH简介闪存模块组织Flash基本结构&#xff08;关系&#xff09;图Flash解锁使用指针访问存储器FLASH操作Flash全擦除Flash页擦除Flash写入 选项字节选项字节操作选项字节擦除选项字节写入 器件电子签名注意闪存控制寄存器一览 前言 本…

Spring Boot 2 入门基础

学习要求 ● 熟悉Spring基础 ● 熟悉Maven使用 环境要求 ● Java8及以上 ● Maven 3.3及以上&#xff1a;https://docs.spring.io/spring-boot/docs/current/reference/html/getting-started.html#getting-started-system-requirements 学习资料 ● 文档地址&#xff1a; htt…

松下MINAS A6B系列旋转电机规格书--A系列

一、松下电机型号的识别方法 二、松下标准型电机型号大全 三、松下电机的规格 四、电机外观 五、XA&#xff0c;XB连接器 六、USB连接器 七、EtherCAT用连接器X2A、X2B 八、IO连接器X4 输入输出信号接口 九、编码器连接器 十、模拟监视器用连接器X7 十一、电源连接器以及端子台…

python-web框架应用程序-Django环境搭建

python-web应用程序-Django环境搭建 一、Django入门 使用Django&#xff08;http://djangoproject.com/&#xff09;来开发一个名为“学习笔记”&#xff08;Learning Log&#xff09;的项目&#xff0c;这是一个在线日志系统&#xff0c;让你能够记录所学习的有关特定主题的知…

DiskCatalogMaker for Mac:您的磁盘目录管理专家

对于需要管理大量磁盘文件的用户来说&#xff0c;DiskCatalogMaker for Mac无疑是一款不可或缺的工具。这款专为Mac用户设计的磁盘目录制作软件&#xff0c;以其简洁的操作界面和强大的功能&#xff0c;帮助您轻松创建和管理磁盘目录。 DiskCatalogMaker支持多种磁盘格式&…

CSS学习笔记之高级教程(一)

1、圆角&#xff1a;border-radius border-radius 属性可以接受一到四个值。规则如下&#xff1a; 四个值 - border-radius: 15px 50px 30px 5px;&#xff08;依次分别用于&#xff1a;左上角、右上角、右下角、左下角&#xff09; 三个值 - border-radius: 15px 50px 30px;&a…

简介有限面积和无限周长

前言 分形理论是一种非常重要的科学概念,它被广泛应用于物理学、数学、生物学等领域。分形理论描述了一种重复自相似的结构,这种结构在不同的尺度上都具有类似的形态。由于分形理论的应用广泛且深远,了解分形理论可以帮助人们更好地理解自然界和人造世界中的现象。 作为一…

增强团队建设和创造力的 6 个敏捷游戏

加入敏捷框架提供了对资源的访问和支持&#xff0c;可以帮助你的组织最大限度地发挥敏捷的优势。它还提供了一个与其他敏捷从业者联系的平台&#xff0c;以共享最佳实践并相互学习。 实践敏捷工作方法可以让团队按照自己的节奏&#xff0c;尽可能多地发挥创造力来追求目标&…

网易狼人杀 设置点击自动发言

我们玩网易狼人杀 刚开始 都会发现 要按住麦克风才能发言 不得不说 相当的麻烦 我们可以点击如下图 右上角这个设置的齿轮 新弹出的设置面板上 勾选这个点击发言 然后 我们只需要 点一下 就可以进入发言状态 然后 再点一下即可停止发言 会方便非常多

游泳时用什么耳机听歌好?精品榜前四游泳耳机揭秘,款款佳品!

游泳时用什么耳机听歌好&#xff1f;这无疑是众多水上运动爱好者的共同疑问。在享受游泳带来的清凉与畅快时&#xff0c;若能伴随着悦耳的音乐&#xff0c;无疑能让整个体验更加完美。然而&#xff0c;市面上的游泳耳机种类繁多&#xff0c;品质各异&#xff0c;如何选择一款既…

网盘拉新项目,一单6.1,分享就能赚钱,人人可做!

前言 有几个圈友在问我&#xff0c;2024年有没有好的副业变现项目&#xff0c;我盘点近期比较热门的项目&#xff0c;比如红包封面、公众号爆文、视频号分成计划、网盘拉新等&#xff0c;最后我给他们的建议是网盘拉新项目。 该项目算比较老了&#xff0c;但确实是一个比较赚钱…

探索Codigger文件管理器(File Explorer)的创新与实用性

在数字时代&#xff0c;文件资源管理器作为桌面环境中不可或缺的一部分&#xff0c;承担着管理文件和文件夹的重要职责。Codigger文件管理器&#xff08;File Explorer&#xff09;以其独特的创新和实用性&#xff0c;为用户提供了高效、便捷的文件管理体验。 Codigger文件管理…

k8s 的 pod 基础 2

一、pod的镜像拉取策略 &#xff08;image PullPolicy&#xff09; Pod 的核心是运行容器&#xff0c;必须指定容器引擎&#xff0c;比如 Docker&#xff0c;启动容器时&#xff0c;需要拉取镜像&#xff0c;k8s 的镜像拉取策略可以由用户指定&#xff1a; 1、IfNotPresent &…

NTLM Relay Gat:自动化NTLM中继安全检测工具

关于NTLM Relay Gat NTLM Relay Gat是一款功能强大的NTLM中继威胁检测工具&#xff0c;该工具旨在利用Impacket工具套件中的ntlmrelayx.py脚本在目标环境中实现NTLM中继攻击风险检测&#xff0c;以帮助研究人员确定目标环境是否能够抵御NTLM中继攻击。 功能介绍 1、多线程支持…

关于如何在 smartforms 中 debug

发现一旦smartforms 复杂起来&#xff0c;Debug的时候就一下子找不到指定位置&#xff0c;所以如何才能最简单的找到指定位置呢 以这个为案例 然后打上断点即可debug

Science Advances|柔性超韧半导体纤维的大规模制备(柔性半导体器件/可穿戴电子/纤维器件/柔性电子)

北京大学 雷霆(Ting Lei)团队,在《Science Advances》上发布了一篇题为“Continuous production of ultratough semiconducting polymer fibers with high electronic performance”的论文。论文内容如下: 一、 摘要 共轭聚合物具有良好的光电特性,但其脆性和机械特性差,…

华大单片机下载报错“Error:Flash Download failed-“Cortex-M4””

Error&#xff1a;Flash Download failed-“Cortex-M4” 问题说明 使用华大单片机HC32F460KETA下载程序时&#xff0c;出现关于M4核的报错&#xff0c;具体如下&#xff1a; 此种情况下代码编译时没有问题的&#xff0c;只是下载时就会显示错误。 解决方法 注意最后一步选默…