1.虚拟机栈
线程私有空间,和线程同一时间创建,保存局部变量,部分结果,参与方法的调用和返回。
StackOverflowError和OutOfMemoryError都会出现。
用-Xss设置栈的深度,会影响StackOverflowError抛出的时间
栈帧保存局部变量表,操作数栈,动态连接方法和返回地址等信息
方法调用 --> 入栈
方法返回 --> 出栈
参数和局部变量多,则栈帧的局部变量表会变大。且局部变量表和性能调优关系最为密切。局部变量表以字为单位进行内存划分,一个字32位。long和double都占2个字,this指针占一个字,其他都占一个字,对于非static方法,虚拟机会将当前对象this作为参数通过局部变量表传递给当前方法。
2.堆
对象和数组分配在堆上。Java堆分为新生代和老年代,新生代又分为Eden,s0,s1。新生代gc会将一个survivor中没有被回收的对象移到另一个survivor中。而在FullGC中新生代对象全部被清空,存活的对象移入老年代
3.方法区
被所有线程共享保存类的元数据:类,静态变量,常量。常量池是方法区的一部分。
方法区也可以被GC,
(1)对常量池的回收
(2)对类元数据的回收
常量池快满的 时候,GC总能回收常量池数据
而对于类的元数据,如使用Javassist生成的字节码类,则不会自动被回收。而HotSpot虚拟机针对此情况,做了设计,在ClassLoader被回收之后,在FullGC时,永久区中的类元数据会被回收。
GC参数:
1.最大堆内存
-Xmx
指新生代和老年代的大小之和的最大值
2.最小堆内存
-Xms
首先会分配-Xms指定的内存大小,并尽可能尝试在这个空间段内运行程序。-Xms内存不够时,JVM才会向操作系统申请更多的内存,直到达到-Xmx。当超过-Xmx,则会报OutOfMemoryError。
-Xms数值过小,JVM为保证系统尽可能在指定范围内运行,会频繁进行GC操作,释放内存。会增加MinorGC和GullGC的次数。
3.设置新生代
-Xmn设置新生代的大小,增加新生代大小会减少老年代的大小,一般设置新生代大小为整个堆空间的1/4到1/3左右
-XX:NewSize设置新生代的初始大小,-XX:MaxNewSize设置新生代的最大值。通常只设置-Xmn已经可以满足大部分的应用的需要。
4.设置持久代
-XX:MaxPermSize设置持久代的最大值,-XX:PermSize设置持久代的初始大小
持久代的大小决定了系统支持定义多少类和常量,对于CGLIB和Javassist等动态字节码生成工具而言,设置合理的持久代大小有利于系统的稳定。
系统的最大类数量与MaxPermSize成正比。一般来说MaxPermSize设置为64M已经可以满足绝大多数应用,如果不够,则可以加到128M。如果依然不够,则可以使用之前的优化方式,将类加载器设置为null自动会回收动态字节码类的生成,使系统健康运行。
5.设置线程栈
-Xss设置线程栈的大小
局部变量分配都需要在栈中分配空间,空间太小,就会导致函数调用深度不够,程序异常;空间过大,则内存成本会上升,系统可以支持的线程总数就会下降。
堆也是向操作系统申请空间的,如果堆过大,则操作系统给栈的空间就会变小,从而减小程序所能支持线程的数量。
由于操作系统内存不够而无法创建新的线程时,导致的OOM异常不是因为堆内存不够导致的,是因为操作系统内存减去堆内存后,剩余的系统内存不够无法创建新的线程,可以尝试减小堆内存,换取更多系统空间。
如果系统需要大量线程并发执行,那么设置一个较小的堆和较小的栈,能够有助于提升系统能够承受的最大线程数目。
LoadClass和Class.ForName的区别
loadClass默认是不去链接这个类的
Class.ForName默认是会去链接这个类,类已经被初始化完成,即static代码块会被执行
Class.ForName应用:如加载MySQL数据库驱动时,要用Class.ForName完成初始化,执行com.mysql.jdbc.Driver中的Static代码块
loadCLass应用:在Spring中大多使用延迟加载,完成对类的加载,使用loadClass来加载类,加快加载速度,把类的实例化留到使用的时候去完成
元空间(MetaSpace)和永久代(PermGen)的区别
元空间使用的是本地/本机 内存
永久代使用的是JVM的内存
使用本地内存的好处:
java.lang.OutOfMemoryError:PermGen space不再存在
本地内存有多大,元空间就有多大,解决了空间不足的问题
**字符串常量池存在永久代中,容易出现性能问题和内存溢出
**类和方法的信息大小难以确定,给永久代的大小指定带来困难
**为GC带来不必要的复杂性
**方便HotSpot和其他JVM如Jrockit的集成
调优参数-Xms -Xmx -Xmn的含义
-Xss:规定了每个线程虚拟机栈(堆栈)的大小
-Xms:堆的初始值
-Xmx:堆能达到的最大值
Java内存模型中堆和栈的区别?
不同JDK版本中的intern()方法的区别 jdk6 vs jdk6+
jdk1.6中仅仅会在常量池中添加对象,1.6+的版本中会在堆中添加对象的引用
1.7以后字符串常量池已经从方法区中移动到了堆中(因为之前常量池存在永久代中,而永久代中的内存空间有限,会造成常量池异常 java.lang.OutOfMemoryError:PermGen space)