答案仅是根据我的印象回答
面试官:来,随便说一下Java虚拟机
我的答案:按调理来简述
1、堆
存几乎所有的对象、数组等。
是gc发生的主要场所。
堆里边分为新生代、老年代,新生代又被分为Survivor 0和Survivor 1,划分这么细是为了垃圾回收。
数据在里边对于各线程是共享的。
会发生OOM(OutOfMemory)
2、虚拟机栈
线程独享,每个线程的局部变量不可被其他线程修改。
主要存放了对象的引用以及八种基本数据类型
会发生StackOverFlowError
和 OutOfMemoryError
StackOverFlowError
:栈溢出,栈内存不可扩展,栈的深度超过了最大深度,就发生错误
OutOfMemoryError:栈内存可以扩展,但是如果无法再申请到空间,就发生错误
3、程序计数器
线程独享,不会发生OOM。
主要作用是1、线程切换时可以知道执行到的地方。2、线程的流程控制,如循环,分支,跳转等。
4、方法区
堆的一个逻辑部分,存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据
方法区也被称为永久代,JDK1.8之后变为元空间,使用直接内存。
5、本地方法栈
和虚拟机栈极为相似,区别于为Native方法服务
6、常量池
JDK1.8后
字符串常量池还在堆
运行时常量池在方法区(由元空间实现)
面试官:简述一下对象创建的过程以及每步都做了什么
1、类加载检查
在常量池中看能否找到这个类的引用
检查这个类是否已经被加载、链接、初始化过
如果没有,就乖乖先加载类
2、分配内存
分配内存主要是两种方法--------指针碰撞、空闲列表
指针碰撞:用于内存整齐的情况。
空闲列表:用于内存不整齐的情况。类似于在垃圾堆里找没垃圾的地方
主要是因为在垃圾回收的时候使用算法不同就会造成内存分为整齐和不整齐的情况。如标记清除算法,标记整理算法
这里分配内存要注意一个并发分配线程不安全的问题。
虚拟机采用CAS+失败重试、TLAB来解决
TLAB:每个线程先在新生区分配内存,如果内存不够用时再去采用CAS+失败重试机制
3、赋零值
保证对象在不赋值时可以直接用,让程序可以访问到属性的类型和零值
4、设置对象头
对象头里边存储什么,就设置什么信息,如:
这个对象是哪个类的实例、如何才能找到类的元数据信息、对象的哈希码、对象的 GC 分代年龄等信息
根据运行状态不同(如偏向锁,轻量级锁,重量级锁),对象头内容不同
5、执行init方法
执行 new 指令之后会接着执行init
方法,按照我们的意愿赋值
如有不对,请及时在评论区补充