题目来源:
图片参考:https://blog.csdn.net/u010737756/article/details/104843152
1.半初始化
(1)new 一个分配空间,划分内存区域,此时m的默认值为 0;
(2)T.init 初始化操作。将 8 赋值给 m;
(3)astore_1 —建立连接,指针 t 指向该内存区域
2.DCL单例(double check lock双重锁机制)
public class DclClass {
private static DclClass INSTANCE;
//构造函数私有化
private DclClass() {}
public static DclClass getInstance() {
//双重检查之一
//先判断是否为空,为空则去拿锁
if(INSTANCE == null) {
//若发现锁不在 则继续等锁
//等到空锁时 需要判断在等待时 是否有另外一个线程改变它
//instance 实例应该是已经创建了 所以需要第二重判断是否为空
synchronized (DclClass.class) {
//双重检查之二
if(INSTANCE == null) {
INSTANCE = new DclClass();
}
}
}
//所以第一重判断可以省去吗
//不可以 第一重判断可以节省效率
return INSTANCE;
}
3. DCL单例 到底需要volatile吗?
volatile的特性:1)禁止指令重新排列
2)线程可见性
答案:当然需要
DCL单例在创造时,第(2)步和第(3)步不能颠倒。
如果 指令重排之后,在 init 初始化之前,先就将 t指向内存分配区域,则此时的m可能还是默认值,还没有被重新赋值。
4. 对象在内存中的存储布局
关于markword的详细介绍:
(1) 锁信息 — 00 0(lock占两位, bias_lock占 1位)
(2) java对象年龄:4位 (所以阈值最高是15)
(3) 31位的对象标识hashCode:
普通对象:
markword—8个字节
class pointer—4个字节
padding—对齐
数组:多一个数组大小的信息
对像头的内存分布(锁信息)
5. 对象怎么定位
(1)句柄方式:
java堆将会划分出来一部分内存去来作为句柄池,reference中存储的就是对象的句柄地址。而句柄中则包含对象实例数据的地址和对象类型数据(如对象的类型,实现的接口、方法、父类、field等)的具体地址信息
(2)直接指针:
6. 对象怎么分配
参考文章:
https://blog.csdn.net/yangsnow_rain_wind/article/details/80434323
https://www.cnblogs.com/BlueStarWei/p/9358757.html
https://www.cnblogs.com/lanmao123/p/10485416.html
(1) 栈上分配:满足逃逸分析之后,在函数调用结束后自行销毁对象,不需要垃圾回收器的介入,有效避免垃圾回收带来的负面影响; 效率高
(1) TLAB(ThreadLocalApplicationBuffer–线程本地分配缓存):分配在堆上(堆是全局共享的)。每个线程会从Eden分配一大块空间,TLAB启用的情况下(默认开启),JVM会为每一个线程分配一块TLAB区域
7. object对象在内存中占用多少字节
参考:https://www.jianshu.com/p/91e398d5d17c
markword 8字节,因为java默认使用了calssPointer压缩,classpointer 4字节,padding 4字节 因此是16字节
如果没开启classpointer默认压缩,markword 8字节,classpointer 8字节,padding 0字节 也是16字节
问题: User (int id,String name) User u = new User(1,‘张三’) 占用多少字节?
markword 8字节,开启classPointer压缩 ,classpointer 4字节,instance data int 4字节,开启普通对象指针压缩 String 4字节 padding 4 一共24字节
8. class对象是在堆区还是方法区
(1)为什么不使用c++对象直接表示java对象