java对象不再使用时赋值null的意义

先看代码

public class TestDemo1 {
    public static void main(String[] args) {
        if (true) {
            byte[] placeHolder = new byte[64 * 1024 * 1024];
            System.out.println(placeHolder.length / 1024);
        }
        System.gc();
    }
}

idea配置gc日志打印

运行上面的代码,载图gc日志

现在我们修改上面的测试代码,将 placeHolder置为null

public class TestDemo1 {
    public static void main(String[] args) {
        if (true) {
            byte[] placeHolder = new byte[64 * 1024 * 1024];
            System.out.println(placeHolder.length / 1024);
            placeHolder = null;
        }
        System.gc();
    }
}

再次运行程序,查看gc日志

由以上载图日志可以明显看到二者差别, 所以不用对象置为null还是很有意义的。

为啥会造成二者的区别呢?

这还得从jvm认定垃圾的机制:可达性分析说起。

说起这个可达性,首先就得说到根,而“本地变量表”恰恰就可以看成是根。

上面两段代码本地变量表是不一样的。

先看第一段代码,就是 placeHolder没有置null的“本地变量表 ”

使用javap -v TestDemo1.class

可以看到 placeHolder还在本地变量表中,而且它占用slot槽1号位置, 所以jvm认为它还是活着的。

然后,我们再看placeHolder = null 这段代码的"本地变量表"的情况,其实它与上面一样,看不出啥差别。

但是如果我们在placeHolder后面再声明一个变量

public class TestDemo1 {
    public static void main(String[] args) {
        if (true) {
            byte[] placeHolder = new byte[64 * 1024 * 1024];
            System.out.println(placeHolder.length / 1024);
        }
        String name = "admin";
        System.gc();
    }
}

可以看到name 这个变量名将 slot槽1号位置占用了,是否可以说明placeHolder没啥用了呢

而且这段代码与 placeHolder = null 的gc日志完全一样。 那么应该可以说明,我们声明的这个String name = "admin"  断开了栈中placeHolder与堆中实例 之间关系。

而placeHolder = null 应该也有这个功能。

总结: 代码离开变量作用域时,并不会自动切断其与堆的联系。

原文地址:https://www.cnblogs.com/z-qinfeng/p/12317084.html

时间: 2024-10-03 09:36:13

java对象不再使用时赋值null的意义的相关文章

fstream对象重复使用时注意clear()的调用

fstream对象重复使用时注意clear()的调用,否则会造成打开第二个文件失败.这是因为一个fstream对象对应磁盘上的一个文件,这种绑定关系在调用open()函数或者构造函数时指定,但有时我们会重复使用同一个fstream对象先后绑定不同文件,在两次绑定中间仅仅调用close()是不够的(当然对close()的调用是必须的,每次打开文件使用完毕都必须关闭文件),因为close()函数并不会重置fstream的条件状态,如果先前的某个操作失败了,那么failbit的状态会一直保留下来,接下

别名现象,java对象之间的相互赋值

请看一下代码 import java.util.*; class book{ static  int c = null; } public static void main(String[] args){ Book book1 = new book(); Book book2 = new book(); book1.c = 6; book2.c = 4; book1 = book2; } 以上,当book1和book2两个对象之间相互赋值的时候,那么当我们调用Book.c的时候,我们就会发现bo

[ Java学习基础 ] Java对象的创建和销毁

类实例化可生成对象,实例方法就是对象方法,实例变量就是对象属性.一个对象的生命周期包括三个阶段:创建.使用和销毁. 创建对象 创建对象包括两个步骤:声明和实例化. 声明 声明对象与声明普通变量没有区别,语法格式如下: type objectName; 其中type是引用类型,即类.接口和数组.示例代码如下: String name; 该语句声明了字符串类型对象name.可以声明并不为对象分配内存空间,而只是分配一个引用. 实例化 实例化过程分为两个阶段:为对象分配内存空间和初始化对象,首先使用n

java对象生命周期

java中一个对象的完整生命周期涉及java平台的很多技术.在创建一个java对象之前,需要先由虚拟机加载该类,然后对该java类进行链接和初始化.初始化完成之后,才创建出该类的对象实例.java对象也有自己的初始化过程,主要通过构造方法完成.当不再有引用指向该对象时,对象占用的内存会在合适的时机被垃圾回收器回收.对象终止机制提供了一种方式在对象被回收之前进行清理工作. java类的链接 虚拟机刚启动时,内部只包含java核心类的相关信息.随着程序的运行,不断有新的java类被加载到虚拟机中.j

Java对象在内存中的状态

可达的/可触及的 Java对象呗创建后,如果被一个或者多个变量引用,那就是可达的,即从根节点可以触及到这个对象. 其实就是从根节点扫描,只要这个对象在引用链中,那就是可触及的. 可恢复的 Java对象不再被任何变量引用就进入了可恢复状态. 在回收该对象之前,该对象的finalize()方法进行资源整理,如果在finalize()方法中重新让变量引用该对象,则该对象再次变为可达状态,否则该对象进入不可达状态. 不可达的: Java对象不被任何变量引用,且系统在调用对象的finalize()方法后依

getContext在谷歌浏览器中,使用时要先加载canvas对象,否则会提示'getContext is null'

<body> <canvas id="myCanvas" width="200" height="100" style="border:1px solid #c3c3c3;"> Your browser does not support the canvas element. </canvas> <script type="text/javascript">

JAVA变量初始化赋值null

 在Java中,null值表示引用不指向任何对象.运行过程中系统发现使用了这样一个引用时·可以立即停止进一步的访问,不会给系统带来任何危险. 1.如果是对象的field的话那么系统在初始化对象的时候会把那些对象的值赋值为null,基本数据类型也有它们的初始值 2.如果是在方法内,那么程序员需要自己给这些变量赋值. class T{   Object obj = null;   Object obj1;//系统在初始化的时候会自动的给他赋值为null } class T{   public voi

实战Java内存泄漏问题分析 -- hazelcast2.0.3使用时内存泄漏 -- 1

公司当年有一个自己缓存集群用户session的Java library,是基于hazlcast2.0.3实现的,最近在customer site集群环境中某个blade报了Out of Memory Exception, 其他blades都正常,马上用jrockit jrcmd命令dump了堆和线程进行分析. printf "##################### heap ##################\n" su -p occas -c "/opt/jrocki

实战Java内存泄漏问题分析 -- hazelcast2.0.3使用时内存泄漏 -- 2

hazelcast 提供了3中方法调用startCleanup: 第一种是在ConcuurentMapManager的构造函数中,通过调用node的executorManager中的ScheduledExecutorService来创建每秒执行一次cleanup操作的线程(代码如下).由于这是ConcuurentMapManager构造函数的代码,所以这种调用startCleanup的操作是默认就会有的. node.executorManager.getScheduledExecutorServ