java虚拟机

Posted 三号小玩家

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了java虚拟机相关的知识,希望对你有一定的参考价值。

jvm简称java虚拟机。jvm使用软件模拟java字节码的指令集。

jvm规范:定义了

        ---Class文件类型

        ---运行时数据

        ---帧栈

        ----虚拟机的启动

        ----虚拟机的指令集

vm指令集:类型转换,出栈入栈操作,运算,流程控制,函数调用。

 

jvm需要对java Library提供以下支持:

--反射java.lang.reflect

--ClassLoader

--初始化class和interface

--安全相关 java.security

--多线程

--弱引用

 

 jvm编译:javap进行反汇编

 

jvm启动的流程:在当前路径和系统版本找jvm.cfg文件,根据配置找到JVM.dll,初始化JVM虚拟机获得JNIEnv接口,找到main方法开始运行。

jvm基本结构:

  pc寄存器:

每个线程拥有一个pc寄存器。

在线程创建时创建。

指向下一条指令的地址。

执行本地方法时,pc的值为undefined。

 

方法区:

保存装载的类信息,通常和永久区关联在一起。

类型的常量池

字段,方法信息

方法字节码

 

java堆:

和程序开发密切相关,new出来的对象都在这里。

应用系统对象都保存在java堆中。

所有线程共享java堆。

对分代GC来说,堆也是分代的。

GC主要工作是区间。

 

java栈:

线程私有。

栈由一系列帧组成(因此栈也叫帧栈)

栈保存一个方法的局部变量,炒作数栈,常量池指针。

每一次方法调用创建一个帧,并压栈。

 栈上分配:

 

Java栈 – 栈上分配,只能分配小对象
小对象(一般几十个bytes),在没有逃逸的情况下,可以直接分配在栈上
直接分配在栈上,可以自动回收,减轻GC压力
大对象或者逃逸对象无法栈上分配

 内存模型:

每个线程有一个工作内存和主存独立

工作内存存放主存中变量值得拷贝

线程共享:

线程总是在自己的本地内存中存储变量,本地内存存储共享内存变量的一个副本,共享内存在主存当中的,共享变量存在时延和误差。

volatile 不能代替锁
一般认为volatile 比锁性能好(不绝对)

选择使用volatile的条件是:
语义是否满足应用
volatile 不是线程安全的

可见性
一个线程修改了变量,其他线程可以立即知道
保证可见性的方法
volatile
synchronized (unlock之前,写变量值回主存)
final(一旦初始化完成,其他线程就可见)

另外参考:http://www.cnblogs.com/43726581Gavin/p/9066080.html

有序性
在本线程内,操作都是有序的
在线程外观察,操作都是无序的。(指令重排 或 主内存同步延时)
指令重排
线程内串行语义
写后读    a = 1;b = a;    写一个变量之后,再读这个位置。
写后写    a = 1;a = 2;    写一个变量之后,再写这个变量。
读后写    a = b;b = 1;    读一个变量之后,再写这个变量。
以上语句不可重排,有依赖性的代码不可以重排
编译器不考虑多线程间的语义
可重排: a=1;b=2;

指令重排原则:

指令重排的基本原则
程序顺序原则:一个线程内保证语义的串行性
volatile规则:volatile变量的写,先发生于读
锁规则:解锁(unlock)必然发生在随后的加锁(lock)前
传递性:A先于B,B先于C 那么A必然先于C
线程的start方法先于它的每一个动作
线程的所有操作先于线程的终结(Thread.join())
线程的中断(interrupt())先于被中断线程的代码
对象的构造函数执行结束先于finalize()方法
解释运行
解释执行以解释方式运行字节码
解释执行的意思是:读一句执行一句
编译运行(JIT)
将字节码编译成机器码
直接执行机器码
运行时编译
编译后性能有数量级的提升
Java代码经过javac编译成class文件(字节码)

class(字节码)文件经过JVM编译成机器码进行解释执行(解释执行)

对于热点代码,JIT(JustInTime)编译器会在运行时将其编译为机器码执行(编译运行)

从上面我们可以看出,在JVM默认的mixed模式下JAVA既不完全是解释运行也不完全是编译运行。

 jvm参数设置:

-Xmx20m -Xms20m -Xmn15m  -XX:+PrintGCDetails -Xloggc:log/gc.log

设置jvm参数并且将参数输出到文件目录:

Java HotSpot(TM) 64-Bit Server VM (25.152-b16) for windows-amd64 JRE (1.8.0_152-b16), built on Sep 14 2017 02:29:36 by "java_re" with MS VC++ 10.0 (VS2010)
Memory: 4k page, physical 8281888k(3109468k free), swap 10247968k(3269608k free)
CommandLine flags: -XX:InitialHeapSize=20971520 -XX:MaxHeapSize=20971520 -XX:MaxNewSize=15728640 -XX:NewSize=15728640 -XX:+PrintGC -XX:+PrintGCDetails -XX:+PrintGCTimeStamps -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:-UseLargePagesIndividualAllocation -XX:+UseParallelGC 
0.231: [GC (Allocation Failure) [PSYoungGen: 11527K->1528K(13824K)] 11527K->1760K(18944K), 0.0018172 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
Heap
 PSYoungGen      total 13824K, used 2800K [0x00000000ff100000, 0x0000000100000000, 0x0000000100000000)
  eden space 12288K, 10% used [0x00000000ff100000,0x00000000ff23e308,0x00000000ffd00000)
  from space 1536K, 99% used [0x00000000ffd00000,0x00000000ffe7e040,0x00000000ffe80000)
  to   space 1536K, 0% used [0x00000000ffe80000,0x00000000ffe80000,0x0000000100000000)
 ParOldGen       total 5120K, used 232K [0x00000000fec00000, 0x00000000ff100000, 0x00000000ff100000)
  object space 5120K, 4% used [0x00000000fec00000,0x00000000fec3a000,0x00000000ff100000)
 Metaspace       used 3436K, capacity 4496K, committed 4864K, reserved 1056768K
  class space    used 373K, capacity 388K, committed 512K, reserved 1048576K

 

 

JVM中的新生代和老年代(Eden空间、两个Survior空间)

 

以上是关于java虚拟机的主要内容,如果未能解决你的问题,请参考以下文章

Android 逆向ART 脱壳 ( DexClassLoader 脱壳 | DexClassLoader 构造函数 | 参考 Dalvik 的 DexClassLoader 类加载流程 )(代码片段

Android 逆向ART 脱壳 ( DexClassLoader 脱壳 | DexClassLoader 构造函数 | 参考 Dalvik 的 DexClassLoader 类加载流程 )(代码片段

Java 虚拟机

理解Java虚拟机体系结构

Java虚拟机 - Class类文件结构

编写Java代码使java虚拟机崩溃