想学习JVM,java虚拟机的底层原理。下面介绍下怎么将Java文件compiler成字节码,然后反编译为二进制查看分析。

一、JavaClass.java文件:

package com.gqz.javasys.principle;

/**
* @ClassName: JavaClass
* @Description: TODO(这里用一句话描述这个类的作用)
* @author ganquanzhong
* @date 2019年11月19日 下午9:43:01
*/
public class JavaClass { private static int CODE_MODE = 96; private String username = "gqzdev"; public void show() {
System.out.println(username+CODE_MODE);
} public static void main(String[] args) {
// TODO Auto-generated method stub
new JavaClass().show();
} }

二、将后缀名为java的源码文件编译成可被JVM可执行的二进制文件:

  找到你要反编译的java文件,cmd输入命令:

javac JavaClass.java

  然后敲回车。如下图:

  用notepad++打开编译后的class文件,会是一串看不懂的乱码:

三、反编译二进制文件文件:

  找到你要反编译的class文件,cmd输入命令:

javap -c -verbose JavaClass.class

  然后敲回车。如下图:

  在反编译后的代码中,可以看到整个class文件执行的过程。

    class常量池类型分类

     class文件结构参照表全集

四、Java字节码指令解释:

Java字节码指令大全

常量入栈指令

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0x01

aconst_null

null值入栈。

0x02

iconst_m1

-1(int)值入栈。

0x03

iconst_0

0(int)值入栈。

0x04

iconst_1

1(int)值入栈。

0x05

iconst_2

2(int)值入栈。

0x06

iconst_3

3(int)值入栈。

0x07

iconst_4

4(int)值入栈。

0x08

iconst_5

5(int)值入栈。

0x09

lconst_0

0(long)值入栈。

0x0a

lconst_1

1(long)值入栈。

0x0b

fconst_0

0(float)值入栈。

0x0c

fconst_1

1(float)值入栈。

0x0d

fconst_2

2(float)值入栈。

0x0e

dconst_0

0(double)值入栈。

0x0f

dconst_1

1(double)值入栈。

0x10

bipush

valuebyte

valuebyte值带符号扩展成int值入栈。

0x11

sipush

valuebyte1

valuebyte2

(valuebyte1 << 8) | valuebyte2 值带符号扩展成int值入栈。

0x12

ldc

indexbyte1

常量池中的常量值(int, float, string reference, object reference)入栈。

0x13

ldc_w

indexbyte1

indexbyte2

常量池中常量(int, float, string reference, object reference)入栈。

0x14

ldc2_w

indexbyte1

indexbyte2

常量池中常量(long, double)入栈。

局部变量值转载到栈中指令

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0x19

(wide)aload

indexbyte

从局部变量indexbyte中装载引用类型值入栈。

0x2a

aload_0

从局部变量0中装载引用类型值入栈。

0x2b

aload_1

从局部变量1中装载引用类型值入栈。

0x2c

aload_2

从局部变量2中装载引用类型值入栈。

0x2d

aload_3

从局部变量3中装载引用类型值入栈。

0x15

(wide)iload

indexbyte

从局部变量indexbyte中装载int类型值入栈。

0x1a

iload_0

从局部变量0中装载int类型值入栈。

0x1b

iload_1

从局部变量1中装载int类型值入栈。

0x1c

iload_2

从局部变量2中装载int类型值入栈。

0x1d

iload_3

从局部变量3中装载int类型值入栈。

0x16

(wide)lload

indexbyte

从局部变量indexbyte中装载long类型值入栈。

0x1e

lload_0

从局部变量0中装载int类型值入栈。

0x1f

lload_1

从局部变量1中装载int类型值入栈。

0x20

lload_2

从局部变量2中装载int类型值入栈。

0x21

lload_3

从局部变量3中装载int类型值入栈。

0x17

(wide)fload

indexbyte

从局部变量indexbyte中装载float类型值入栈。

0x22

fload_0

从局部变量0中装载float类型值入栈。

0x23

fload_1

从局部变量1中装载float类型值入栈。

0x24

fload_2

从局部变量2中装载float类型值入栈。

0x25

fload_3

从局部变量3中装载float类型值入栈。

0x18

(wide)dload

indexbyte

从局部变量indexbyte中装载double类型值入栈。

0x26

dload_0

从局部变量0中装载double类型值入栈。

0x27

dload_1

从局部变量1中装载double类型值入栈。

0x28

dload_2

从局部变量2中装载double类型值入栈。

0x29

dload_3

从局部变量3中装载double类型值入栈。

0x32

aaload

从引用类型数组中装载指定项的值。

0x2e

iaload

从int类型数组中装载指定项的值。

0x2f

laload

从long类型数组中装载指定项的值。

0x30

faload

从float类型数组中装载指定项的值。

0x31

daload

从double类型数组中装载指定项的值。

0x33

baload

从boolean类型数组或byte类型数组中装载指定项的值(先转换为int类型值,后压栈)。

0x34

caload

从char类型数组中装载指定项的值(先转换为int类型值,后压栈)。

0x35

saload

从short类型数组中装载指定项的值(先转换为int类型值,后压栈)。

将栈顶值保存到局部变量中指令

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0x3a

(wide)astore

indexbyte

将栈顶引用类型值保存到局部变量indexbyte中。

0x4b

astroe_0

将栈顶引用类型值保存到局部变量0中。

0x4c

astore_1

将栈顶引用类型值保存到局部变量1中。

0x4d

astore_2

将栈顶引用类型值保存到局部变量2中。

0x4e

astore_3

将栈顶引用类型值保存到局部变量3中。

0x36

(wide)istore

indexbyte

将栈顶int类型值保存到局部变量indexbyte中。

0x3b

istore_0

将栈顶int类型值保存到局部变量0中。

0x3c

istore_1

将栈顶int类型值保存到局部变量1中。

0x3d

istore_2

将栈顶int类型值保存到局部变量2中。

0x3e

istore_3

将栈顶int类型值保存到局部变量3中。

0x37

(wide)lstore

indexbyte

将栈顶long类型值保存到局部变量indexbyte中。

0x3f

lstore_0

将栈顶long类型值保存到局部变量0中。

0x40

lstore_1

将栈顶long类型值保存到局部变量1中。

0x41

lstore_2

将栈顶long类型值保存到局部变量2中。

0x42

lstroe_3

将栈顶long类型值保存到局部变量3中。

0x38

(wide)fstore

indexbyte

将栈顶float类型值保存到局部变量indexbyte中。

0x43

fstore_0

将栈顶float类型值保存到局部变量0中。

0x44

fstore_1

将栈顶float类型值保存到局部变量1中。

0x45

fstore_2

将栈顶float类型值保存到局部变量2中。

0x46

fstore_3

将栈顶float类型值保存到局部变量3中。

0x39

(wide)dstore

indexbyte

将栈顶double类型值保存到局部变量indexbyte中。

0x47

dstore_0

将栈顶double类型值保存到局部变量0中。

0x48

dstore_1

将栈顶double类型值保存到局部变量1中。

0x49

dstore_2

将栈顶double类型值保存到局部变量2中。

0x4a

dstore_3

将栈顶double类型值保存到局部变量3中。

0x53

aastore

将栈顶引用类型值保存到指定引用类型数组的指定项。

0x4f

iastore

将栈顶int类型值保存到指定int类型数组的指定项。

0x50

lastore

将栈顶long类型值保存到指定long类型数组的指定项。

0x51

fastore

将栈顶float类型值保存到指定float类型数组的指定项。

0x52

dastore

将栈顶double类型值保存到指定double类型数组的指定项。

0x54

bastroe

将栈顶boolean类型值或byte类型值保存到指定boolean类型数组或byte类型数组的指定项。

0x55

castore

将栈顶char类型值保存到指定char类型数组的指定项。

0x56

sastore

将栈顶short类型值保存到指定short类型数组的指定项。

wide指令

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0xc4

wide

使用附加字节扩展局部变量索引(iinc指令特殊)。

通用(无类型)栈操作指令

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0x00

nop

空操作。

0x57

pop

从栈顶弹出一个字长的数据。

0x58

pop2

从栈顶弹出两个字长的数据。

0x59

dup

复制栈顶一个字长的数据,将复制后的数据压栈。

0x5a

dup_x1

复制栈顶一个字长的数据,弹出栈顶两个字长数据,先将复制后的数据压栈,再将弹出的两个字长数据压栈。

0x5b

dup_x2

复制栈顶一个字长的数据,弹出栈顶三个字长的数据,将复制后的数据压栈,再将弹出的三个字长的数据压栈。

0x5c

dup2

复制栈顶两个字长的数据,将复制后的两个字长的数据压栈。

0x5d

dup2_x1

复制栈顶两个字长的数据,弹出栈顶三个字长的数据,将复制后的两个字长的数据压栈,再将弹出的三个字长的数据压栈。

0x5e

dup2_x2

复制栈顶两个字长的数据,弹出栈顶四个字长的数据,将复制后的两个字长的数据压栈,再将弹出的四个字长的数据压栈。

0x5f

swap

交换栈顶两个字长的数据的位置。Java指令中没有提供以两个字长为单位的交换指令。

类型转换指令

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0x86

i2f

将栈顶int类型值转换为float类型值。

0x85

i2l

将栈顶int类型值转换为long类型值。

0x87

i2d

将栈顶int类型值转换为double类型值。

0x8b

f2i

将栈顶float类型值转换为int类型值。

0x8c

f2l

将栈顶float类型值转换为long类型值。

0x8d

f2d

将栈顶float类型值转换为double类型值。

0x88

l2i

将栈顶long类型值转换为int类型值。

0x89

l2f

将栈顶long类型值转换为float类型值。

0x8a

l2d

将栈顶long类型值转换double类型值。

0x8e

d2i

将栈顶double类型值转换为int类型值。

0x90

d2f

将栈顶double类型值转换为float类型值。

0x8f

d2l

将栈顶double类型值转换为long类型值。

0x91

i2b

将栈顶int类型值截断成byte类型,后带符号扩展成int类型值入栈。

0x92

i2c

将栈顶int类型值截断成char类型值,后带符号扩展成int类型值入栈。

0x93

i2s

将栈顶int类型值截断成short类型值,后带符号扩展成int类型值入栈。

整数运算

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0x60

iadd

将栈顶两int类型数相加,结果入栈。

0x64

isub

将栈顶两int类型数相减,结果入栈。

0x68

imul

将栈顶两int类型数相乘,结果入栈。

0x6c

idiv

将栈顶两int类型数相除,结果入栈。

0x70

irem

将栈顶两int类型数取模,结果入栈。

0x74

ineg

将栈顶int类型值取负,结果入栈。

0x61

ladd

将栈顶两long类型数相加,结果入栈。

0x65

lsub

将栈顶两long类型数相减,结果入栈。

0x69

lmul

将栈顶两long类型数相乘,结果入栈。

0x6d

ldiv

将栈顶两long类型数相除,结果入栈。

0x71

lrem

将栈顶两long类型数取模,结果入栈。

0x75

lneg

将栈顶long类型值取负,结果入栈。

0x84

(wide)iinc

indexbyte

constbyte

将整数值constbyte加到indexbyte指定的int类型的局部变量中。

浮点运算

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0x62

fadd

将栈顶两float类型数相加,结果入栈。

0x66

fsub

将栈顶两float类型数相减,结果入栈。

0x6a

fmul

将栈顶两float类型数相乘,结果入栈。

0x6e

fdiv

将栈顶两float类型数相除,结果入栈。

0x72

frem

将栈顶两float类型数取模,结果入栈。

0x76

fneg

将栈顶float类型值取反,结果入栈。

0x63

dadd

将栈顶两double类型数相加,结果入栈。

0x67

dsub

将栈顶两double类型数相减,结果入栈。

0x6b

dmul

将栈顶两double类型数相乘,结果入栈。

0x6f

ddiv

将栈顶两double类型数相除,结果入栈。

0x73

drem

将栈顶两double类型数取模,结果入栈。

0x77

dneg

将栈顶double类型值取负,结果入栈。

逻辑运算——移位运算

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0x78

ishl

左移int类型值。

0x79

lshl

左移long类型值。

0x7a

ishr

算术右移int类型值。

0x7b

lshr

算术右移long类型值。

0x7c

iushr

逻辑右移int类型值。

0x7d

lushr

逻辑右移long类型值。

逻辑运算——按位布尔运算

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0x73

iand

对int类型按位与运算。

0x7f

land

对long类型的按位与运算。

0x80

ior

对int类型的按位或运算。

0x81

lor

对long类型的按位或运算。

0x82

ixor

对int类型的按位异或运算。

0x83

lxor

对long类型的按位异或运算。

控制流指令——条件跳转指令

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0x99

ifeq

branchbyte1

branchbyte2

若栈顶int类型值为0则跳转。

0x9a

ifne

branchbyte1

branchbyte2

若栈顶int类型值不为0则跳转。

0x9b

iflt

branchbyte1

branchbyte2

若栈顶int类型值小于0则跳转。

0x9e

ifle

branchbyte1

branchbyte2

若栈顶int类型值小于等于0则跳转。

0x9d

ifgt

branchbyte1

branchbyte2

若栈顶int类型值大于0则跳转。

0x9c

ifge

branchbyte1

branchbyte2

若栈顶int类型值大于等于0则跳转。

0x9f

if_icmpeq

branchbyte1

branchbyte2

若栈顶两int类型值相等则跳转。

0xa0

if_icmpne

branchbyte1

branchbyte2

若栈顶两int类型值不相等则跳转。

0xa1

if_icmplt

branchbyte1

branchbyte2

若栈顶两int类型值前小于后则跳转。

0xa4

if_icmple

branchbyte1

branchbyte2

若栈顶两int类型值前小于等于后则跳转。

0xa3

if_icmpgt

branchbyte1

branchbyte2

若栈顶两int类型值前大于后则跳转。

0xa2

if_icmpge

branchbyte1

branchbyte2

若栈顶两int类型值前大于等于后则跳转。

0xc6

ifnull

branchbyte1

branchbyte2

若栈顶引用值为null则跳转。

0xc7

ifnonnull

branchbyte1

branchbyte2

若栈顶引用值不为null则跳转。

0xa5

if_acmpeq

branchbyte1

branchbyte2

若栈顶两引用类型值相等则跳转。

0xa6

if_acmpne

branchbyte1

branchbyte2

若栈顶两引用类型值不相等则跳转。

控制流指令——比较指令

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

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入栈。

控制流指令——无条件跳转指令

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0xa7

goto

branchbyte1

branchbyte2

无条件跳转到指定位置。

0xc8

goto_w

branchbyte1

branchbyte2

branchbyte3

branchbyte4

无条件跳转到指定位置(宽索引)。

控制流指令——表跳转指令

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0xaa

tableswitch

<0-3bytepad>

defaultbyte1

defaultbyte2

defaultbyte3

defaultbyte4

lowbyte1

lowbyte2

lowbyte3

lowbyte4

highbyte1

highbyte2

highbyte3

highbyte4

jump offsets...

通过索引访问跳转表,并跳转。

0xab

lookupswitch

<0-3bytepad>

defaultbyte1

defaultbyte2

defaultbyte3

defaultbyte4

npairs1

npairs2

npairs3

npairs4

match offsets

通过键值访问跳转表,并跳转。

控制流指令——异常和finally

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0xbf

athrow

抛出异常。

0xa8

jsr

branchbyte1

branchbyte2

跳转到子例程序。

0xc9

jsr_w

branchbyte1

branchbyte2

branchbyte3

branchbyte4

跳转到子例程序(宽索引)。

0xa9

(wide)ret

indexbyte

返回子例程序。

对象操作指令

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0xbb

new

indexbyte1

indexbyte2

创建新的对象实例。

0xc0

checkcast

indexbyte1

indexbyte

类型强转。

0xc1

instanceof

indexbyte1

indexbyte2

判断类型。

0xb4

getfield

indexbyte1

indexbyte2

获取对象字段的值。

0xb5

putfield

indexbyte1

indexbyte2

给对象字段赋值。

0xb2

getstatic

indexbyte1

indexbyte2

获取静态字段的值。

0xb3

putstatic

indexbyte1

indexbyte2

给静态字段赋值。

数组操作指令

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0xbc

newarray

atype

创建type类型的数组。

0xbd

anewarray

indexbyte1

indexbyte2

创建引用类型的数组。

0xbe

arraylength

获取一维数组的长度。

0xc5

multianewarray

indexbyte1

indexbyte2

dimension

创建dimension维度的数组。

方法调用指令

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0xb7

invokespecial

indexbyte1

indexbyte2

编译时方法绑定调用方法。

0xb6

invokevirtual

indexbyte1

indexbyte2

运行时方法绑定调用方法。

0xb8

invokestatic

indexbyte1

indexbyte2

调用静态方法。

0xb9

invokeinterface

indexbyte1

indexbyte2

count

0

调用接口方法。

方法返回指令

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0xac

ireturn

返回int类型值。

0xad

lreturn

返回long类型值。

0xae

freturn

返回float类型值。

0xaf

dreturn

返回double类型值。

0xb0

areturn

返回引用类型值。

0xb1

return

void函数返回。

线程同步指令

指令码

操作码(助记符)

操作数

描述(栈指操作数栈)

0xc2

monitorenter

进入并获得对象监视器。

0xc3

monitorexit

释放并退出对象监视器。

JVM学习-环境构建的更多相关文章

  1. 第1章 Python学习环境构建目录

    第1章 引子 第1.1节 学习环境搭建 第1.2节 Python学习环境的使用 第2章 Python编程基础知识 第2.1节 简单的Python数据类型.变量赋值及输入输出 第2.2节 Python的 ...

  2. JVM学习笔记——内存结构篇

    JVM学习笔记--内存结构篇 在本系列内容中我们会对JVM做一个系统的学习,本片将会介绍JVM的内存结构部分 我们会分为以下几部分进行介绍: JVM整体介绍 程序计数器 虚拟机栈 本地方法栈 堆 方法 ...

  3. JVM学习(4)——全面总结Java的GC算法和回收机制

    俗话说,自己写的代码,6个月后也是别人的代码……复习!复习!复习!涉及到的知识点总结如下: 一些JVM的跟踪参数的设置 Java堆的分配参数 -Xmx 和 –Xms 应该保持一个什么关系,可以让系统的 ...

  4. JVM学习(3)——总结Java内存模型

    俗话说,自己写的代码,6个月后也是别人的代码……复习!复习!复习!涉及到的知识点总结如下: 为什么学习Java的内存模式 缓存一致性问题 什么是内存模型 JMM(Java Memory Model)简 ...

  5. Java虚拟机JVM学习05 类加载器的父委托机制

    Java虚拟机JVM学习05 类加载器的父委托机制 类加载器 类加载器用来把类加载到Java虚拟机中. 类加载器的类型 有两种类型的类加载器: 1.JVM自带的加载器: 根类加载器(Bootstrap ...

  6. Java虚拟机JVM学习03 连接过程:验证、准备、解析

    Java虚拟机JVM学习03 连接过程:验证.准备.解析 类被加载后,就进入连接阶段. 连接就是将已经读入到内存的类的二进制数据合并到虚拟机的运行时环境中去. 连接阶段三个步骤:验证.准备和解析. 类 ...

  7. java之jvm学习笔记六-十二(实践写自己的安全管理器)(jar包的代码认证和签名) (实践对jar包的代码签名) (策略文件)(策略和保护域) (访问控制器) (访问控制器的栈校验机制) (jvm基本结构)

    java之jvm学习笔记六(实践写自己的安全管理器) 安全管理器SecurityManager里设计的内容实在是非常的庞大,它的核心方法就是checkPerssiom这个方法里又调用 AccessCo ...

  8. java之jvm学习笔记四(安全管理器)

    java之jvm学习笔记四(安全管理器) 前面已经简述了java的安全模型的两个组成部分(类装载器,class文件校验器),接下来学习的是java安全模型的另外一个重要组成部分安全管理器. 安全管理器 ...

  9. JVM学习001通过实例总结Java虚拟机的运行机制

    JVM学习(1)——通过实例总结Java虚拟机的运行机制-转载http://www.cnblogs.com/kubixuesheng/p/5199200.html 文章转载自:http://www.c ...

随机推荐

  1. Android中点击按钮启动另一个Activity以及Activity之间传值

    场景 点击第一个Activity中的按钮,启动第二个Activity,关闭第二个Activity,返回到第一个Activity. 在第一个Activity中给第二个Activity传递值,第二个Act ...

  2. FIB表与RIB表的区别与联系

    RIB (route information base) 和 FIB (forwarding information base),又称Ip路由表 和 CEF表,它们之间的关系可以用下面这张图片来高度概 ...

  3. adams技巧汇总

    T+左键        平动模型 R+左键        旋转模型 Z+左键        动态缩放 F或Ctrl+F     以最大比例全面显示模型 S+左键        沿着垂直于屏幕的轴线旋转 ...

  4. PLSQL-Initialization error

    1. 情况描述 安装完PLSQL之后,在写入数据库连接信息后,连接数据库,报Initialization error错误 (1)未安装Oracle数据库,无相应的oci.dll文件 (2)已安装数据库 ...

  5. Electron – 项目报错整理(打包~2): electron-packager踩坑

  6. SpringBoot 教程之发送邮件

    目录   1. 简介  2. API  3. 配置  4. 实战  5. 示例源码  6. 参考资料 1. 简介 Spring Boot 收发邮件最简便方式是通过 spring-boot-starte ...

  7. HashMap初始化容量过程

    集合是Java开发日常开发中经常会使用到的,而作为一种典型的K-V结构的数据结构,HashMap对于Java开发者一定不陌生.在日常开发中,我们经常会像如下方式以下创建一个HashMap: Map&l ...

  8. Redis-03-集群搭建

    基于redis-3.2.4的Redis-Cluster集群搭建 原理 Redis 集群采用了P2P的模式,完全去中心化.Redis 把所有的 Key 分成了 16384 个 slot,每个 Redis ...

  9. 数据类型(8种)和运算符——Java

    一.什么是标识符,它有什么作用(重点掌握) 1. 标识符指的是 标识符是用户编程时使用的名字,用于给变量.常量.函数.语句块等命名,以建立起名称与使用之间的关系.标识符可由任何字母数字字符串形成. 2 ...

  10. 【daily】日常所遇 - 页面A嵌套页面B

    因为懒,所以直接在http://www.runoob.com写demo测试了. (1)iframe嵌套 虽然可能到现在这中方式都很常用,但是你baidu/google一下会发现.很多人都并不提倡用if ...