JVM00_面试官对类加载器子系统、运行时数据区、内存分布、执行引擎的灵虚拷问,你能坚持到第几问?(八)

④②. 垃圾回收是否会涉及到虚拟机栈?


  • ①. 不会;垃圾回收只会涉及到方法区和堆中,方法区和堆也会存在溢出的可能


  • ②. 程序计数器,只记录运行下一行的地址,不存在溢出和垃圾回收


  • ③. 虚拟机栈和本地方法栈,都是只涉及压栈和出栈,可能存在栈溢出,不存在垃圾回收


④③. 方法中定义的局部变量是否线程安全?


  • 如果局部变量在内部产生并在内部消亡的,那就是线程安全的


④④. 什么情况下会发生栈内存溢出?


  • ①. 局部数组过大。当函数内部的数组过大时,有可能导致堆栈溢出


  • ②. 递归调用层次太多。递归函数在运行时会执行压栈操作,当压栈次数太多时,也会导致堆栈溢出


④⑤. 说说堆和栈的区别?


①.从GC、OOM、*Error的角度


[栈中不存在GC,当固定大小的栈会发生*Error,动态的会发生OOM。堆中GC、OOM、*Error都存在]


②. 从堆栈的执行效率[栈的效率高于堆]


③. 内存大小,数据结构

[堆的空间比栈的大一般,栈是一种FIFO先进后出的模型。堆中结构复杂,可以有链表、数组等]


④. 栈管运行,堆管存储


④⑥. 简单谈谈你对堆的理解?(共享|垃圾回收)


①. Java堆区在JVM启动的时候即被创建,其空间大小也是确定的。是Jvm管理最大的一块内存空间


②. 所有的线程共享Java堆,在这里还可以划分线程私有的缓冲区(Thread Local Allocation Buffer,TLAB)


③. 在方法结束后,堆中的对象不会马上被移除,仅仅在垃圾收集的时候才有被移除

(注意:一个进程就是一个JVM实例,一个进程中包含多个线程)


④. 所有的线程共享Java堆,在这里还可以划分线程私有的缓冲区(Thread Local Allocation Buffer, TLAB)


④⑦. 堆里面的分区:Eden,survival(from+to),老年代,各自的特点


①. 在JDK1.7中分为: 新生代+老年代+永久代 | 在JDK1.8中分为: 新生代+老年代+元空间


②. 新生代:伊甸园区、幸存者S0、S1(8:1:1),几乎所有的Java对象都是在Eden区被new出来的,


绝大部分的Java对象的销毁都在新生代进行了;IBM公司的专门研究表明,新生代中 80% 的对象都是"朝生夕死"的


老年代:另外一类对象的生命周期却非常长,在某些极端的情况下还能够与JVM的生命周期保持一致


新生代:老年代=1:2


④⑧. 堆大小通过什么参数设置?


①. -Xms:初始内存(默认为物理内存的1/64)


②. -Xmx:最大内存(默认为物理内存的1/4)


③. -XX:NewRatio=2


-XX:SurvivorRatio


-XX:HandlePromotionFailure:空间分配担保


-Xmn:设置新生代最大内存大小,一般使用默认值就可以了


④⑨. 初始堆大小和最大堆大小一样,问这样有什么好处?


通常会将-Xms和-Xmx两个参数配置相同的值,其目的是为了能够在java垃圾回收机制清理完堆区后不需要重新分隔计算堆区的大小,从而提升性能


⑤0. 对象在堆中的分配规则?


①. 优先分配到Eden


②. 大对象直接分配到老年(尽量避免程序中出现过多的大对象)


③. 长期存活的对象分配到老年代


④. 动态对象年龄判断(如果Survivor 区中相同年龄的所有对象大小的总和大于Survivor空间的一半,年龄大于或等于该年龄对象可以直接进入老年代,无须等到MaxTenurningThreshold中要求的年龄


⑤. 空间分配担保 -XX:HandlePromotionFailure

(JDK6 Update24之后规则变为只有老年代的连续空间大于新生代对象总大小或者历次晋身的平均大小就会进行Minor GC,否则将进行Full GC)


上一篇:window下pip install Scrapy报错解决方案


下一篇:C#中通过Selenium定位标签的问题