JVM概述
虚拟机:模拟某种计算机体系结构,执行特定指令集的软件。
两大类:
1.系统虚拟机(Virtual Box,VMware):运行系统的
2.程序虚拟机(JVM,.NET CLR(C#),P-Code):运行程序的。
Java虚拟机架构
HotSpot虚拟机
运行较高的代码直接编译成二进制,存储在内存中,提高效率。我们的java就是用的HotSpot
类加载
Java类加载器(Java Classloader)是Java运行时环境的一部分,负责动态加载Java类到Java虚拟机的内存空间中。
类通常是按需加载,即第一次使用该类时才加载。由于有了类加载器,Java运行时系统不需要知道文件与文件系统。
-verbose:class:显示类加载的过程。
父子层级关系不是继承的关系,而是上下级的关系
//系统类加载器
//1第一种方式
ClassLoader appsLoader = ClassLoader.getSystemClassLoader();
System.out.println(appsLoader.toString());
//2.第二种方式
ClassLoader appLoader = Test01.class.getClassLoader();
System.out.println(appLoader.toString());
//扩展类加载器
//第一种方式
ClassLoader extLoader = appsLoader.getParent();
System.out.println(extLoader.toString());
//第二种方式
ClassLoader extLoader1= SunEC.class.getClassLoader();
System.out.println(extLoader1.toString());
//启动类加载器
System.out.println(extLoader1.getParent());
类加载过程
利用类加载器读取文件
//使用类加载器加载属性文件,以src包为根,不包含src
InputStream inputStream=ClassLoader.getSystemClassLoader().getResourceAsStream("qf/test/test02/user.properties");
Properties properties = new Properties();
properties.load(inputStream);
properties.list(System.out);
inputStream.close();
Properties properties = new Properties();
//以当前类所在的目录为根
InputStream resourceAsStream1 = TestJar.class.getResourceAsStream("user.properties");
//也可以写成这样
InputStream resourceAsStream = TestJar.class.getResourceAsStream("/qf/test/test02/user.properties");
properties.load(resourceAsStream);
properties.list(System.out);
resourceAsStream.close();
运行jar包,java -jar 名.jar
运行时数据区
程序计数器
虚拟机栈和本地方法栈
栈
栈帧
赋值操作也不是原子操作,先从常量池中取出放入操作数栈中,然后出栈放入局部变量表中
局部变量表
存储数据
操作数栈
执行操作和返回结果都在栈中。
动态连接(实现多态)
方法返回地址
堆(heap)
栈不需要垃圾回收。堆需要垃圾回收
“+”是启动,“-”是禁止
survivor区只能有一个存储数据。From和To相互转换,首先会存储在From区,之后相互转换。大小为一般为5M,所以空闲的内存大小会少5M。
堆-对象分配策略
Java对象
Mark Word:用于存储对象自身的运行时数据,如HashCode(哈希码),GC分代年龄,锁状态标志,线程持有的锁,偏向线程ID,偏向时间戳等,这部分数据的长度可以为32bit或者64bit
类型指针(Klass Pointer):对象指向它的类型元数据的指针,Java虚拟机通过这个指针来确定该对象是那个类的实例。
如果对象是一个Java数组,那在对象头中还必须有一块用于记录数组长度的数据。
方法区和运行时常量池
方法区
类对象在堆里面
运行时常量池
字符串常量就在运行时常量池中。
HotSpot方法区实现的变迁
直接内存
直接内存并不是虚拟机运行时数据区的一部分,也不是Java虚拟机规范中定义的内存区域,但是这部分内存也被频繁地使用。
在JDK1.4的NIO中已经出现直接内存的使用
也可能出现OutOfMemoryError异常。
其他空间
TLAB是线程私有的。是存放在Eden区,很小的一部分。
逃逸分析(运行的时候进行逃逸分析)
第一个是逃逸,第二个不是。
垃圾收集
释放垃圾占用的空间,防止内存溢出或者内存泄漏。
垃圾:没有任何引用指向的对象,成为垃圾。
垃圾判定算法:
1.引用计数算法
2.可达性分析算法
Root对象
Java对象引用
finalization
常见垃圾回收算法
1.标记-清除算法(Mark-Sweep)
需要记录可用空间。
2.标记-整理算法(Mark-Compact)
下次直接使用指针碰撞分配空间
3.复制算法
适用于堆中的年轻代。
分代收集机制
垃圾回收器:
1.Serial收集器(串行收集器)
ParNew收集器
3.Parallel Scavenge收集器
Serial Old 收集器
Parallel Old 收集器
CMS收集器
初始标记和重新标记时间短。但是并发标记和并发清除允许并发的
G1收集器(JDK1.9默认)