【JVM原理探索】class字节码指令操作介绍(下)

简介: 【JVM原理探索】class字节码指令操作介绍(下)

线程栈介绍


  • 每一个线程都有一个保存帧的栈。在每一个方法调用的时候创建一个帧。一个帧主要包括三个核心部分:操作栈,局部变量数组,和一个对当前方法所属类的常量池的引用


  • 局部变量数组也被称之为局部变量表,它包含了方法的参数,也用于保存一些局部变量的值。


  • 参数值得存放总是在局部变量数组的index0开始的。如果当前帧是由构造函数或者实例方法创建的,那么该对象引用(this)将会存放在location0处,然后才开始存放其余的参数


  • 局部变量表的大小由编译时决定,同时也依赖于局部变量的数量和一些方法的大小


  • 操作栈是一个(LIFO)栈,用于压入和取出值,其大小也在编译时决定某些opcode指令将值压入操作栈(dup),其余的opcode指令将操作数取出栈(pop、swap)


  • 使用它们后再把结果压入栈。操作栈也用于接收从方法中返回的值



JVM指令集大全


下面三个为保留操作码,是留给虚拟机内部使用的。


  • 0xca breakpoint  调试时的断点标记
  • 0xfe impdep1    为特定软件而预留的语言后门
  • 0xff impdep2    为特定硬件而预留的语言后门


下面是所有的JVM指令集,可能个别会有出入,可以参看下《Java虚拟机规范 (JavaSE8版)》


  • 指令码 助记符    说明
  • 0x00 nop        无操作
  • 0x01 aconst_null 将null推送至栈顶
  • 0x02 iconst_m1    将int型-1推送至栈顶
  • 0x03 iconst_0    将int型0推送至栈顶
  • 0x04 iconst_1    将int型1推送至栈顶
  • 0x05 iconst_2    将int型2推送至栈顶
  • 0x06 iconst_3    将int型3推送至栈顶
  • 0x07 iconst_4    将int型4推送至栈顶
  • 0x08 iconst_5    将int型5推送至栈顶

  • 0x09 lconst_0    将long型0推送至栈顶
  • 0x0a lconst_1    将long型1推送至栈顶

  • 0x0b fconst_0    将float型0推送至栈顶
  • 0x0c fconst_1    将float型1推送至栈顶
  • 0x0d fconst_2    将float型2推送至栈顶

  • 0x0e dconst_0    将double型0推送至栈顶
  • 0x0f dconst_1    将double型1推送至栈顶

  • 0x10 bipush    将单字节的常量值(-128~127)推送至栈顶
  • 0x11 sipush    将一个短整型常量值(-32768~32767)推送至栈顶

  • 0x12 ldc    将int, float或String型常量值从常量池中推送至栈顶
  • 0x13 ldc_w    将int, float或String型常量值从常量池中推送至栈顶(宽索引)
  • 0x14 ldc2_w    将long或double型常量值从常量池中推送至栈顶(宽索引)

  • 0x15 iload    将指定的int型本地变量推送至栈顶
  • 0x16 lload    将指定的long型本地变量推送至栈顶

  • 0x17 fload    将指定的float型本地变量推送至栈顶
  • 0x18 dload    将指定的double型本地变量推送至栈顶
  • 0x19 aload    将指定的引用类型本地变量推送至栈顶

  • 0x1a iload_0    将第一个int型本地变量推送至栈顶
  • 0x1b iload_1    将第二个int型本地变量推送至栈顶
  • 0x1c iload_2    将第三个int型本地变量推送至栈顶
  • 0x1d iload_3    将第四个int型本地变量推送至栈顶

  • 0x1e lload_0    将第一个long型本地变量推送至栈顶
  • 0x1f lload_1    将第二个long型本地变量推送至栈顶
  • 0x20 lload_2    将第三个long型本地变量推送至栈顶
  • 0x21 lload_3    将第四个long型本地变量推送至栈顶

  • 0x22 fload_0    将第一个float型本地变量推送至栈顶
  • 0x23 fload_1    将第二个float型本地变量推送至栈顶
  • 0x24 fload_2    将第三个float型本地变量推送至栈顶
  • 0x25 fload_3    将第四个float型本地变量推送至栈顶

  • 0x26 dload_0    将第一个double型本地变量推送至栈顶
  • 0x27 dload_1    将第二个double型本地变量推送至栈顶
  • 0x28 dload_2    将第三个double型本地变量推送至栈顶
  • 0x29 dload_3    将第四个double型本地变量推送至栈顶

  • 0x2a aload_0    将第一个引用类型本地变量推送至栈顶
  • 0x2b aload_1    将第二个引用类型本地变量推送至栈顶
  • 0x2c aload_2    将第三个引用类型本地变量推送至栈顶
  • 0x2d aload_3    将第四个引用类型本地变量推送至栈顶

  • 0x2e iaload    将int型数组指定索引的值推送至栈顶
  • 0x2f laload    将long型数组指定索引的值推送至栈顶
  • 0x30 faload    将float型数组指定索引的值推送至栈顶
  • 0x31 daload    将double型数组指定索引的值推送至栈顶

  • 0x32 aaload    将引用型数组指定索引的值推送至栈顶
  • 0x33 baload    将boolean或byte型数组指定索引的值推送至栈顶
  • 0x34 caload    将char型数组指定索引的值推送至栈顶
  • 0x35 saload    将short型数组指定索引的值推送至栈顶

  • 0x36 istore    将栈顶int型数值存入指定本地变量
  • 0x37 lstore    将栈顶long型数值存入指定本地变量
  • 0x38 fstore    将栈顶float型数值存入指定本地变量
  • 0x39 dstore    将栈顶double型数值存入指定本地变量
  • 0x3a astore    将栈顶引用型数值存入指定本地变量

  • 0x3b istore_0    将栈顶int型数值存入第一个本地变量
  • 0x3c istore_1    将栈顶int型数值存入第二个本地变量
  • 0x3d istore_2    将栈顶int型数值存入第三个本地变量
  • 0x3e istore_3    将栈顶int型数值存入第四个本地变量

  • 0x3f lstore_0    将栈顶long型数值存入第一个本地变量
  • 0x40 lstore_1    将栈顶long型数值存入第二个本地变量
  • 0x41 lstore_2    将栈顶long型数值存入第三个本地变量
  • 0x42 lstore_3    将栈顶long型数值存入第四个本地变量

  • 0x43 fstore_0    将栈顶float型数值存入第一个本地变量
  • 0x44 fstore_1    将栈顶float型数值存入第二个本地变量
  • 0x45 fstore_2    将栈顶float型数值存入第三个本地变量
  • 0x46 fstore_3    将栈顶float型数值存入第四个本地变量

  • 0x47 dstore_0    将栈顶double型数值存入第一个本地变量
  • 0x48 dstore_1    将栈顶double型数值存入第二个本地变量
  • 0x49 dstore_2    将栈顶double型数值存入第三个本地变量
  • 0x4a dstore_3    将栈顶double型数值存入第四个本地变量

  • 0x4b astore_0    将栈顶引用型数值存入第一个本地变量
  • 0x4c astore_1    将栈顶引用型数值存入第二个本地变量
  • 0x4d astore_2    将栈顶引用型数值存入第三个本地变量
  • 0x4e astore_3    将栈顶引用型数值存入第四个本地变量

  • 0x4f iastore    将栈顶int型数值存入指定数组的指定索引位置
  • 0x50 lastore    将栈顶long型数值存入指定数组的指定索引位置
  • 0x51 fastore    将栈顶float型数值存入指定数组的指定索引位置
  • 0x52 dastore    将栈顶double型数值存入指定数组的指定索引位置
  • 0x53 aastore    将栈顶引用型数值存入指定数组的指定索引位置
  • 0x54 bastore    将栈顶boolean或byte型数值存入指定数组的指定索引位置
  • 0x55 castore    将栈顶char型数值存入指定数组的指定索引位置
  • 0x56 sastore    将栈顶short型数值存入指定数组的指定索引位置

  • 0x57 pop     将栈顶数值弹出 (数值不能是long或double类型的)
  • 0x58 pop2    将栈顶的一个(long或double类型的)或两个数值弹出(其它)

  • 0x59 dup     复制栈顶数值并将复制值压入栈顶
  • 0x5a dup_x1    复制栈顶数值并将两个复制值压入栈顶
  • 0x5b dup_x2    复制栈顶数值并将三个(或两个)复制值压入栈顶
  • 0x5c dup2    复制栈顶一个(long或double类型的)或两个(其它)数值并将复制值压入栈顶
  • 0x5d dup2_x1    复制栈顶的一个或两个值,将其插入栈顶那两个或三个值的下面
  • 0x5e dup2_x2    复制栈顶的一个或两个值,将其插入栈顶那两个、三个或四个值的下面

  • 0x5f swap    将栈最顶端的两个数值互换(数值不能是long或double类型的)

  • 0x60 iadd    将栈顶两int型数值相加并将结果压入栈顶
  • 0x61 ladd    将栈顶两long型数值相加并将结果压入栈顶
  • 0x62 fadd    将栈顶两float型数值相加并将结果压入栈顶
  • 0x63 dadd    将栈顶两double型数值相加并将结果压入栈顶

  • 0x64 isub    将栈顶两int型数值相减并将结果压入栈顶
  • 0x65 lsub    将栈顶两long型数值相减并将结果压入栈顶
  • 0x66 fsub    将栈顶两float型数值相减并将结果压入栈顶
  • 0x67 dsub    将栈顶两double型数值相减并将结果压入栈顶

  • 0x68 imul    将栈顶两int型数值相乘并将结果压入栈顶
  • 0x69 lmul    将栈顶两long型数值相乘并将结果压入栈顶
  • 0x6a fmul    将栈顶两float型数值相乘并将结果压入栈顶
  • 0x6b dmul    将栈顶两double型数值相乘并将结果压入栈顶

  • 0x6c idiv    将栈顶两int型数值相除并将结果压入栈顶
  • 0x6d ldiv    将栈顶两long型数值相除并将结果压入栈顶
  • 0x6e fdiv    将栈顶两float型数值相除并将结果压入栈顶
  • 0x6f ddiv    将栈顶两double型数值相除并将结果压入栈顶

  • 0x70 irem    将栈顶两int型数值作取模运算并将结果压入栈顶
  • 0x71 lrem    将栈顶两long型数值作取模运算并将结果压入栈顶
  • 0x72 frem    将栈顶两float型数值作取模运算并将结果压入栈顶
  • 0x73 drem    将栈顶两double型数值作取模运算并将结果压入栈顶

  • 0x74 ineg    将栈顶int型数值取负并将结果压入栈顶
  • 0x75 lneg    将栈顶long型数值取负并将结果压入栈顶
  • 0x76 fneg    将栈顶float型数值取负并将结果压入栈顶
  • 0x77 dneg    将栈顶double型数值取负并将结果压入栈顶

  • 0x78 ishl    将int型数值左移位指定位数并将结果压入栈顶
  • 0x79 lshl    将long型数值左移位指定位数并将结果压入栈顶
  • 0x7a ishr    将int型数值右(符号)移位指定位数并将结果压入栈顶
  • 0x7b lshr    将long型数值右(符号)移位指定位数并将结果压入栈顶

  • 0x7c iushr    将int型数值右(无符号)移位指定位数并将结果压入栈顶
  • 0x7d lushr    将long型数值右(无符号)移位指定位数并将结果压入栈顶

  • 0x7e iand    将栈顶两int型数值作“按位与”并将结果压入栈顶
  • 0x7f land    将栈顶两long型数值作“按位与”并将结果压入栈顶
  • 0x80 ior     将栈顶两int型数值作“按位或”并将结果压入栈顶
  • 0x81 lor     将栈顶两long型数值作“按位或”并将结果压入栈顶
  • 0x82 ixor    将栈顶两int型数值作“按位异或”并将结果压入栈顶
  • 0x83 lxor    将栈顶两long型数值作“按位异或”并将结果压入栈顶
  • 0x84 iinc    将指定int型变量增加指定值(i++, i--, i+=2)

  • 0x85 i2l     将栈顶int型数值强制转换成long型数值并将结果压入栈顶
  • 0x86 i2f     将栈顶int型数值强制转换成float型数值并将结果压入栈顶
  • 0x87 i2d     将栈顶int型数值强制转换成double型数值并将结果压入栈顶
  • 0x88 l2i     将栈顶long型数值强制转换成int型数值并将结果压入栈顶
  • 0x89 l2f     将栈顶long型数值强制转换成float型数值并将结果压入栈顶
  • 0x8a l2d     将栈顶long型数值强制转换成double型数值并将结果压入栈顶
  • 0x8b f2i     将栈顶float型数值强制转换成int型数值并将结果压入栈顶
  • 0x8c f2l     将栈顶float型数值强制转换成long型数值并将结果压入栈顶
  • 0x8d f2d     将栈顶float型数值强制转换成double型数值并将结果压入栈顶
  • 0x8e d2i     将栈顶double型数值强制转换成int型数值并将结果压入栈顶
  • 0x8f d2l     将栈顶double型数值强制转换成long型数值并将结果压入栈顶
  • 0x90 d2f     将栈顶double型数值强制转换成float型数值并将结果压入栈顶
  • 0x91 i2b     将栈顶int型数值强制转换成byte型数值并将结果压入栈顶
  • 0x92 i2c     将栈顶int型数值强制转换成char型数值并将结果压入栈顶
  • 0x93 i2s     将栈顶int型数值强制转换成short型数值并将结果压入栈顶

  • 0x94 lcmp    比较栈顶两long型数值大小,并将结果(1,0,-1)压入栈顶
  • 0x95 fcmpl    比较栈顶两float型数值大小,并将结果(1,0,-1)压入栈顶;当其中一个数值为NaN时,将-1压入栈顶
  • 0x96 fcmpg    比较栈顶两float型数值大小,并将结果(1,0,-1)压入栈顶;当其中一个数值为NaN时,将1压入栈顶
  • 0x97 dcmpl    比较栈顶两double型数值大小,并将结果(1,0,-1)压入栈顶;当其中一个数值为NaN时,将-1压入栈顶
  • 0x98 dcmpg    比较栈顶两double型数值大小,并将结果(1,0,-1)压入栈顶;当其中一个数值为NaN时,将1压入栈顶
  • 0x99 ifeq    当栈顶int型数值等于0时跳转
  • 0x9a ifne    当栈顶int型数值不等于0时跳转
  • 0x9b iflt    当栈顶int型数值小于0时跳转
  • 0x9c ifge    当栈顶int型数值大于等于0时跳转
  • 0x9d ifgt    当栈顶int型数值大于0时跳转
  • 0x9e ifle    当栈顶int型数值小于等于0时跳转
  • 0x9f if_icmpeq    比较栈顶两int型数值大小,当结果等于0时跳转
  • 0xa0 if_icmpne    比较栈顶两int型数值大小,当结果不等于0时跳转
  • 0xa1 if_icmplt    比较栈顶两int型数值大小,当结果小于0时跳转
  • 0xa2 if_icmpge    比较栈顶两int型数值大小,当结果大于等于0时跳转
  • 0xa3 if_icmpgt    比较栈顶两int型数值大小,当结果大于0时跳转
  • 0xa4 if_icmple    比较栈顶两int型数值大小,当结果小于等于0时跳转
  • 0xa5 if_acmpeq    比较栈顶两引用型数值,当结果相等时跳转
  • 0xa6 if_acmpne    比较栈顶两引用型数值,当结果不相等时跳转
  • 0xa7 goto    无条件跳转
  • 0xa8 jsr     跳转至指定16位offset位置,并将jsr下一条指令地址压入栈顶
  • 0xa9 ret     返回至本地变量指定的index的指令位置(一般与jsr, jsr_w联合使用)
  • 0xaa tableswitch    用于switch条件跳转,case值连续(可变长度指令)
  • 0xab lookupswitch    用于switch条件跳转,case值不连续(可变长度指令)
  • 0xac ireturn    从当前方法返回int
  • 0xad lreturn    从当前方法返回long
  • 0xae freturn    从当前方法返回float
  • 0xaf dreturn    从当前方法返回double
  • 0xb0 areturn    从当前方法返回对象引用
  • 0xb1 return    从当前方法返回void
  • 0xb2 getstatic    获取指定类的静态域,并将其值压入栈顶
  • 0xb3 putstatic    为指定的类的静态域赋值
  • 0xb4 getfield    获取指定类的实例域,并将其值压入栈顶
  • 0xb5 putfield    为指定的类的实例域赋值

  • 0xb6 invokevirtual    调用实例方法
  • 0xb7 invokespecial    调用超类构造方法,实例初始化方法,私有方法
  • 0xb8 invokestatic    调用静态方法
  • 0xb9 invokeinterface 调用接口方法
  • 0xba invokedynamic  调用动态链接方法

  • 0xbb new     创建一个对象,并将其引用值压入栈顶
  • 0xbc newarray    创建一个指定原始类型(如int, float, char…)的数组,并将其引用值压入栈顶
  • 0xbd anewarray    创建一个引用型(如类,接口,数组)的数组,并将其引用值压入栈顶
  • 0xbe arraylength 获得数组的长度值并压入栈顶

  • 0xbf athrow    将栈顶的异常抛出
  • 0xc0 checkcast    检验类型转换,检验未通过将抛出ClassCastException
  • 0xc1 instanceof 检验对象是否是指定的类的实例,如果是将1压入栈顶,否则将0压入栈顶

  • 0xc2 monitorenter    获得对象的锁,用于同步方法或同步块
  • 0xc3 monitorexit    释放对象的锁,用于同步方法或同步块

  • 0xc4 wide    扩大本地变量索引的宽度
  • 0xc5 multianewarray 创建指定类型和指定维度的多维数组(执行该指令时,操作栈中必须包含各维度的长度值),并将其引用值压入栈顶
  • 0xc6 ifnull    为null时跳转
  • 0xc7 ifnonnull    不为null时跳转
  • 0xc8 goto_w    无条件跳转
  • 0xc9 jsr_w    跳转至指定32位offset位置,并将jsr_w下一条指令地址压入栈顶
  • 0xca breakpoint  调试时的断点标记
  • 0xfe impdep1    为特定软件而预留的语言后门
  • 0xff impdep2    为特定硬件而预留的语言后门




相关文章
|
28天前
|
Rust 安全 Java
JVM原理与实现——Synchronized关键字
在多线程Java程序中,`Synchronized`关键字用于确保线程安全。本文深入探讨其工作原理,通过分析字节码`monitorenter`和`monitorexit`,解释JVM如何实现同步机制。文章展示了`Synchronized`方法的编译结果,并详细解析了轻量锁和重度锁的实现过程,包括Mark Word的状态变化及CAS操作的应用。最后简要介绍了`ObjectMonitor::enter()`函数在获取重度锁时的作用。
JVM原理与实现——Synchronized关键字
|
3月前
|
存储 监控 算法
美团面试:说说 G1垃圾回收 底层原理?说说你 JVM 调优的过程 ?
尼恩提示: G1垃圾回收 原理非常重要, 是面试的重点, 大家一定要好好掌握
美团面试:说说 G1垃圾回收 底层原理?说说你 JVM 调优的过程  ?
|
3月前
|
SQL 缓存 Java
JVM知识体系学习三:class文件初始化过程、硬件层数据一致性(硬件层)、缓存行、指令乱序执行问题、如何保证不乱序(volatile等)
这篇文章详细介绍了JVM中类文件的初始化过程、硬件层面的数据一致性问题、缓存行和伪共享、指令乱序执行问题,以及如何通过`volatile`关键字和`synchronized`关键字来保证数据的有序性和可见性。
46 3
|
3月前
|
小程序 Oracle Java
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
这篇文章是关于JVM基础知识的介绍,包括JVM的跨平台和跨语言特性、Class文件格式的详细解析,以及如何使用javap和jclasslib工具来分析Class文件。
68 0
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
|
3月前
|
前端开发 Java 应用服务中间件
JVM进阶调优系列(1)类加载器原理一文讲透
本文详细介绍了JVM类加载机制。首先解释了类加载器的概念及其工作原理,接着阐述了四种类型的类加载器:启动类加载器、扩展类加载器、应用类加载器及用户自定义类加载器。文中重点讲解了双亲委派机制,包括其优点和缺点,并探讨了打破这一机制的方法。最后,通过Tomcat的实际应用示例,展示了如何通过自定义类加载器打破双亲委派机制,实现应用间的隔离。
|
2月前
|
缓存 Prometheus 监控
Elasticsearch集群JVM调优设置合适的堆内存大小
Elasticsearch集群JVM调优设置合适的堆内存大小
518 1
|
3月前
|
存储 安全 Java
jvm 锁的 膨胀过程?锁内存怎么变化的
【10月更文挑战第3天】在Java虚拟机(JVM)中,`synchronized`关键字用于实现同步,确保多个线程在访问共享资源时的一致性和线程安全。JVM对`synchronized`进行了优化,以适应不同的竞争场景,这种优化主要体现在锁的膨胀过程,即从偏向锁到轻量级锁,再到重量级锁的转变。下面我们将详细介绍这一过程以及锁在内存中的变化。
51 4
|
28天前
|
存储 Java 程序员
【JVM】——JVM运行机制、类加载机制、内存划分
JVM运行机制,堆栈,程序计数器,元数据区,JVM加载机制,双亲委派模型
|
1月前
|
存储 监控 算法
深入探索Java虚拟机(JVM)的内存管理机制
本文旨在为读者提供对Java虚拟机(JVM)内存管理机制的深入理解。通过详细解析JVM的内存结构、垃圾回收算法以及性能优化策略,本文不仅揭示了Java程序高效运行背后的原理,还为开发者提供了优化应用程序性能的实用技巧。不同于常规摘要仅概述文章大意,本文摘要将简要介绍JVM内存管理的关键点,为读者提供一个清晰的学习路线图。
|
2月前
|
Java
JVM内存参数
-Xmx[]:堆空间最大内存 -Xms[]:堆空间最小内存,一般设置成跟堆空间最大内存一样的 -Xmn[]:新生代的最大内存 -xx[use 垃圾回收器名称]:指定垃圾回收器 -xss:设置单个线程栈大小 一般设堆空间为最大可用物理地址的百分之80