1.详细jvm内存模型
根据JVM规范,JVM内存模型分为虚拟机栈、堆、方法区、程序计数器、本地方法栈五个部分。
JDK1.8将永久代替换成元空间有什么变化?
(1)永久代存储的数据:符号引用转义到了native heap;字面量转移到了java heap;类的静态变量转移到了java heap
(2)Metaspace(元空间)存储的是类的元数据信息
(3)元空间的本质和永久代类似,都是对JVM规范中方法区的实现。不过元空间与永久代之间的最大区别在于:元空间并不在虚拟机中,而是使用本地内存。
(4)将永久代替换成元空间的好处:
a.字符串存在永久代中,容易出现性能问题和内存溢出。
b.永久代会为GC带来不必要的复杂度,并且回收效率低。
2.什么情况下会出现内存溢出和内存泄漏?
内存泄漏的原因很简单:对象一直被引用,但是对象不会被使用。
常见内存泄漏的例子:定义一个HashSet集合,使用循环向集合中添加Object对象,每次添加到集合后将Object对象设置为null。对象虽然为空,但是set还维护着对象的引用,gc不会回收这些对象。
解决方法:将set设置为null,这时引用将不会被维护,gc会回收这些引用。
参考资料:Java的内存泄漏
内存溢出的原因:
(1)内存泄漏导致堆栈内存不断增大,从而引发溢出。
(2)大量的jar,class文件加载,装载类的空间不够导致溢出。
(3)操作大量的对象导致内存溢出。
(4)nio直接操作内存,内存过大导致溢出。
解决方法:
(1)查看程序是否存在内存泄漏问题。
(2)设置参数加大JVM内存空间。
(3)代码中是否存在死循环或循环产生过多重复的对象。
(4)查看是否使用nio直接操作内存。
参考资料:
java内存溢出的情况解决方法 - goodboy321 - 博客园
3.简单谈谈线程栈
JVM规范让每个java线程拥有自己独立的线程栈,也就是java方法的调用栈。
当方法调用的时候,会产生一个栈帧。栈帧保存在虚拟机栈中,栈帧存储了方法的局部变量表、操作数栈、动态连接和方法返回地址等信息。
线程运行过程中,只有一个栈帧处于活跃状态,称为“当前活跃栈帧”,当前活动栈帧始终是虚拟机栈的栈顶元素。
通过jstack工具可以查看线程状态。
参考资料:
栈帧、局部变量表、操作数栈 - Bruce-水桶妖 - ITeye博客