Object的内存布
使用JavaAgent测试Object的大小
对象大小(64位机)
观察虚拟机配置
java -XX:+PrintCommandLineFlags -version
对象的内存布局分为两种 普通对象和数组对象
普通对象
- 对象头:markword 8 hospot 中叫 markword 占8个字节
- ClassPointer指针:-XX:+UseCompressedClassPointers 为4字节 不开启为8字节 t = new T() 出来对象后 有一个指针 t , 这个指针指向 T.class
- 实例数据
- long int String ….
- 引用类型:-XX:+UseCompressedOops 为4字节 不开启为8字节
Oops Ordinary Object Pointers
- Padding对齐,8的倍数 读取时按照块来读取的 ,目的是为了提升效率 ,对齐后对象大小是8的倍数
数组对象
- 对象头:markword 8
- ClassPointer指针同上
- 数组长度:4字节
- 数组数据
- 对齐 8的倍数
如何观察Object大小?
原理:java不像其他语言可以动态获取大小,需要自己实现一个Agent (Java的一个机制)
Agent:JVM在加载到Class之前,中间可以有一个Agent代理截获Class中的内容,也可以任意修改,通过这个机制获取Object的大小,如下是Agent实现的过程
实验
- 新建项目ObjectSize (1.8)
- 创建文件ObjectSizeAgent
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15import java.lang.instrument.Instrumentation;
public class ObjectSizeAgent {
private static Instrumentation inst;
// premain 格式是固定的
// 这个方法是虚拟机自己调用的 JVM会传入一个 Instrumentation 保存到Agent里面
public static void premain(String agentArgs, Instrumentation _inst) {
inst = _inst;
}
// 通过保存JVM的 Instrumentation 调用 里面的 getObjectSize 方法
public static long sizeOf(Object o) {
return inst.getObjectSize(o);
}
} - src目录下创建META-INF/MANIFEST.MF 注意Premain-Class这行必须是新的一行(回车 + 换行),确认idea不能有任何错误提示
1
2
3Manifest-Version: 1.0
Created-By: chase.com
Premain-Class: com.chase.jvm.agent.ObjectSizeAgent // main 方法运行之前 的一个Class - 打包jar文件
- 在需要使用该Agent Jar的项目中引入该Jar包
project structure - project settings - library 添加该jar包 - 运行时需要该Agent Jar的类,加入参数: 指定Agent
1
-javaagent:C:\work\ijprojects\ObjectSize\out\artifacts\ObjectSize_jar\ObjectSize.jar
- 如何使用该类:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38package com.chase.jvm.c3_jmm;
import com.chase.jvm.agent.ObjectSizeAgent;
public class T03_SizeOfAnObject {
public static void main(String[] args) {
// 16 字节 对象头8 + class指针8压缩后成 4
// 结果:8 + 4 + Padding对齐 = 16
System.out.println(ObjectSizeAgent.sizeOf(new Object()));
// 24字节 对象头8 + class指针8---4 + 数组长度 4 数组内容 没内容= 0字节
// 结果:8 + 4 + 4 + 0 + Padding(4) = 20
System.out.println(ObjectSizeAgent.sizeOf(new int[] {}));
// 8 头 + classPointer 4 + int 4 + String引用 4(压缩后)+ byte 1 + byte 1
// + Object 4(压缩后) + byte 1
// 结果:8+4+4+4+4+1+1+4+1 + Padding(1) = 32
System.out.println(ObjectSizeAgent.sizeOf(new P()));
}
// 一个Object 占用多少个字节
// --- 运行配置
// -XX:+UseCompressedClassPointers 对class指针压缩 默认开启
// -XX:+UseCompressedOops 对普通象指针压缩 默认开启
// Oops = ordinary object pointers 普通对象指针
private static class P {
//8 _markword 对象头 8 个字节
//4 _oop指针 class 指针 8 压缩后4
int id; //4
String name; //4 引用类型,正常是8个字节 压缩后变成4
int age; //4
byte b1; //1
byte b2; //1
Object o; //4
byte b3; //1
}
}
Hotspot开启内存压缩的规则(64位机)
- 4G以下,直接砍掉高32位
- 4G - 32G,默认开启内存压缩 ClassPointers Oops
- 32G,压缩无效,使用64位
内存并不是越大越好(^-^)
IdentityHashCode的问题
当一个对象计算过identityHashCode之后,不能进入偏向锁状态
https://cloud.tencent.com/developer/article/1480590https://cloud.tencent.com/developer/article/1484167
https://cloud.tencent.com/developer/article/1485795
https://cloud.tencent.com/developer/article/1482500
对象定位
t 怎么找到 T() 和 T.class
T t = new T()
- 句柄池 —- 优点:效率高 有一个橘句柄池的东西 存了两个指针一个指向对象,一个指向 Class
- 直接指针 ——优点:GC 回收快 t 指向对象,对象指向 Tlass
不同虚拟机实现的机制不一样 HosPot 是第二种
Object的内存布
http://yoursite.com/post/2c4437f3.html/