回顾
上一篇说到了JVM的内存区域划分
线程私有(程序计数器、虚拟机栈、本地方法栈)
线程共享(堆内存、方法区)
以及 从加载一个类 到 执行方法的流程
对象的分配与引用
上一篇说到,在执行某个方法的时候,如果 遇到 XXX xx = new XXX() ,则会在堆内存分配这个实例对象的内存空间
同时会让当前方法的栈帧内的 xxx 局部变量去指向这个 XXX 对象的堆内存地址
接下来,通过局部变量 xxx 引用的实例对象,去执行它的某个方法,完成业务逻辑
方法执行完毕之后会咋样
如果方法执行完毕,则会让当前调用方法对应的栈帧从Java虚拟机栈中出栈
一旦方法出栈,对应方法的 各种 局部变量,引用,都没有了
也就是说,没有 引用 指向堆内存的 实例对象了
创建的Java对象都是占空间的
对象没被引用,放在内存不是占空间吗
而内存又是有限的
如果内存释放不妥当,会触发 OOM 的
一般我们会在机器上启动一个Java系统,机器的内存资源是有限的,比如 4 个 G
然后我们的 Java 系统,本质也是一个 JVM 进程,它负责运行我们写好的代码
那么,这个JVM本身也是要占用机器的内存的,比如 2 G
我们在 JVM 堆中创建的 Java 对象,其实本质也是会占用 JVM 的内存资源的
那么创建一个 Java 对象 需要占用多少空间呢
一个对象对内存空间的占用,主要分为 2 块
1、对象本身的一些信息
2、对象实例变量作为数据占用的空间
比如对象头
如果是在 64 位 linux 系统上,会占 64 字节
如果有个 int 类型 的实例变量,会占 4 字节
如果是 long 类型的 实例变量,会占 8 字节
如果是 数组,Map ,集合之类的,那就会占更多的内存空间了
不需要的对象,该怎么处理
垃圾回收机制
如果一个实例对象,没有任何一个方法的局部变量指向它,也没有任何一个静态变量、常量指向它
那么垃圾回收线程,就会把没人引用的对象被清理掉,进而释放内存空间