JVM

  • JVM的内存划分
  • 类加载器

四种类加载器:

  1. 启动类加载器:C++编写,在Java中看不到
  2. 扩展类加载器:ExtClassLoader
  3. 应用类加载器:AppicationClassLoader
  4. 自定义类加载器:创建一个类继承java.lang.ClassLoader,定制类加载方式

关系:启动类加载器是扩展类加载器的父加载器,扩展类加载器是应用类加载器的父加载器.

双亲委派模型:

优点:保证了类的唯一性,避免类的重复加载,父加载器加载了某个类,子加载器就不加载了,全类名是唯一标识

安全性考虑,比如String等Jre自带的核心Api,由最父类的启动类加载器加载,不由它的子类加载器恶意替换.

  • 方法栈

堆栈先进后出,后进先出,像是一个弹簧,压栈,弹栈概念,方法每次被调用,都会产生一个栈帧,方法内的局部变量,方法释放,变量也释放

栈帧的结构:

局部变量表:方法执行时的参数、方法体内声明的局部变量

操作数栈:存储中间运算结果,是一个临时存储空间

帧数据区:保存访问常量池指针,异常处理表

操作数栈的存在,保存一个临时存储空间,例如数值交换int a = 5 和int b = 6 交换

栈溢出异常(StackOverflowErro):

例如一个没有退出机制的递归方法,只占用不结束,一直产生栈帧,一直在申请空间,把栈空间挤满,栈空间一直不释放,一直在堆积,把栈内存中的空间耗尽

某一个线程抛出『栈溢出异常』,会导致其他线程也崩溃吗?

不会,线程对栈内存空间的使用方式是彼此隔离的。每个线程都是在自己独享的空间内运行,反过来也可以说,这个空间是当前线程私有的。

工作机制:

新new出来的对象最先放在新生代的伊甸区,

伊甸区的使用空间达到一定的值后触发Minor GC垃圾回收,

没有被这个小GC回收的会成为幸存者,去往幸存者区,

幸存者区有from和to2个区,(谁空谁为to区,意味着这2个区是互相彼此交换的)

from区快满了就去往了to区,此时from变成了空成为了to区,而to区就变成了from区

如果一个对象经过了15次GC还没有被垃圾回收,那么就会去往老年代区保存,

如果幸存者区满了,即是没有15次GC,这个对象也会去往老年代区保存

说明:

eden区存储的主要是短暂临时,生命周期很短的对象

幸存者区是一个中间地带

老年代区是存放生命周期很长的对象,例如IOC管理的对象(Controller,Service,Mapper等等),线程池对象,数据库连接池对象等等...

不被回收是因为栈中一直有个变量指引着这个地址,相当于有一根线连接起来了,如果这个线断了,就是不指向这个对象的地址了,那么就会被GC垃圾回收机制回收,老年代区的对象就是一直有这根指引着

堆溢出异常(OutOfMemeory):

包括两种Java.heap.space和PermGen space,前者是一直在new Object而且不回收,超出了堆内存

后者PermGen space翻译过来也很好理解,permanent generation永久代的问题,是永久代(方法去)加载的时候,类实在太多了,就导致方法区直接溢出了

GC的使用

为什么要GC:

程序运行中,会产生很多对象,如果不处理垃圾对象,那么一直累加会导致内存耗尽,程序崩溃,GC就是处理不使用的垃圾对象,释放内存

如何找出垃圾对象:

引用计数法(不靠谱,如果有循环引用的对象,不能够标记到)

可达性分析(从GC Roots对象出发,不可达的对象就是要清理的对象)

GC的算法:

基本算法

引用计数法:使用引用标记法标记对象被引用的次数,引用加1,删除减1,当引用计数为0 则GC

标记清除法:当堆的有效内存即将耗尽的时候,暂时挂起程序(stop the world),从根对象开始遍历所有的对象,然后再清楚所有没有被标记到的对象

标记压缩法:标记清除法的升级,同样当堆有效内存即将耗尽,暂时挂起,从根对象遍历所有对象,然后中间多一步压缩, 移动所有的可达对象到堆内存的同一个区域中,使他们紧凑的排列在一起,从而将所有非可达对象释放出来的空闲内存都集中在一起,通过这样的方式来达到减少内存碎片的目的。

复制算法:复制算法的核心就是,将原有的内存空间一分为二,每次只用其中的一块,在垃圾回收时,将正在使用的对象复制到另一个内存空间中,并依次排列,然后将该内存空间清空,交换两个内存的角色,完成垃圾的回收。

综合算法

分代算法:前面介绍了多种回收算法,每一种算法都有自己的优点也有缺点,谁都不能替代谁,所以根据垃圾回收对象的特点进行选择,才是明智的。分代算法其实就是这样的,根据回收对象的特点进行选择新生代使用复制算法,老年代使用标记清除或者标记压缩方法

分区算法:有点类似微服务那种分布式,把堆空间划分为多个不同的小区间,每个小区间独立使用独立回收,这样可以控制一次回收多个区间,并不像标记清除和压缩方法一样,需要把程序全部挂起一段时间

JVM的常用参数:

-Xms 堆内存的初始化大小(建议初始化大小和最大值一样,这样就不需要多次提交申请)

-Xmx 堆内存的最大值

-Xmn 新生代内存的大小

-XX:PermSize 永久代的大小

-XX:MaxPermSize 永久代的初始化大小

你可能感兴趣的:(JVM)