给不用的对象设置null的意义

本文通过三个实验探讨了Java中大对象内存回收的影响因素。测试表明,将不再使用的对象设置为null有助于垃圾回收,尤其是在存在大对象且后续有耗时操作的情况下。此外,代码块的作用域限制也能加速内存释放。理解这些机制对于优化Java应用的内存使用至关重要。
摘要由CSDN通过智能技术生成

前言:是否需要把不用的对象设置为null?

1、开始写代码测试(所有测试都要加上以下指令)

jvm参数-Xms20m -Xmx20m -Xmn10m -XX:+PrintGCDetails -XX:SurvivorRatio=8 -XX:PretenureSizeThreshold=2097152

简单解释一下:

  • -Xms20m -Xmx20m这两个指令限制堆内存固定为20m不允许扩容
  • -Xmn10m代表分配给新生代的总内存为10m
  • -XX:SurvivorRatio=8代表Eden区和Survivor的比例8:1,即新生代被分为3部分,分别8m,1m,1m
  • XX:PretenureSizeThreshold=2097152,这个指令用的比较少,在虚拟机中,普通对象都在新生代分配内存,但是大对象是直接在老年代分配,至于多大算大对象,就是这个参数来设置的,我设置的是2m用来测试(2097152 =2 * 1024 * 1024 ),设置2m是方便我测试,保证我在下面代码设置1m的MB_1对象,内存是在新生代分配,而不是直接进入老年代
  • -XX:+PrintGCDetails打印垃圾回收日志
1.1 第一次测试,直接创建一个512kb的数组,调用回收
  public static void main(String[] args) throws InterruptedException {
        byte[] KB_512 = new byte[1 * 1024 * 512];
        System.gc();
    }

结果:

GC (System.gc()) [PSYoungGen: 2229K->992K(9216K)] 2229K->1008K(19456K), 0.0027934 secs] [Times: user=0.01 sys=0.00, real=0.01 secs]
[GC (System.gc()) [PSYoungGen: 2229K->992K(9216K)] 2229K->1000K(19456K), 0.0014906 secs] [Times: user=0.01 sys=0.00, real=0.00 secs] 

多次运行测试,垃圾回收差不多都是2229K->992K,一定要多次运行,因为System.gc()并不是100%触发到此对象的回收

1.2 第二次测试,把KB_512设置null
  public static void main(String[] args) throws InterruptedException {
        byte[] KB_512 = new byte[1 * 1024 * 512];
        KB_512 = null;
        System.gc();
    }

结果:

[GC (System.gc()) [PSYoungGen: 2063K->416K(9216K)] 2063K->424K(19456K), 0.0016337 secs] [Times: user=0.00 sys=0.00, real=0.01 secs] 
[GC (System.gc()) [PSYoungGen: 1899K->480K(9216K)] 1899K->488K(19456K), 0.0027343 secs] [Times: user=0.01 sys=0.00, real=0.00 secs] 

测试多次,发现新生代从2063K->416K (每次测试有一点误差,差别不大),确实比不设置null回收更多

1.3 第三次测试,把本地变量放到代码快
   public static void main(String[] args) throws InterruptedException {
        {
            byte[] KB_512 = new byte[1 * 1024 * 512];
        }
        byte[] temp = new byte[0];
        System.gc();
    }

结果:

[GC (System.gc()) [PSYoungGen: 1899K->464K(9216K)] 1899K->472K(19456K), 0.0041090 secs] [Times: user=0.01 sys=0.00, real=0.00 secs] 
GC (System.gc()) [PSYoungGen: 1899K->480K(9216K)] 1899K->488K(19456K), 0.0031194 secs] [Times: user=0.01 sys=0.00, real=0.01 secs] 

测试结果和第二次大致一样

2、结论

经过以上三次测试的出结论如下:

  1. 当有大对象KB_512使用完后,把其设置为null,是值得是把它在栈中的本地变量那个引用值设置为null,让其失去对堆内存中new byte[1 * 1024 * 512]的引用,当没有其他变量 引用到new byte[1 * 1024 * 512];时,这个new byte[1 * 1024 * 512];对象在垃圾回收触发时是有机会被回收的,注意有机会被回收,不等于这次回收一定会被回收,可能是下次gc时回收,也可能是下下次。

    另外补充一点:栈内存是不用回收的,用完自动释放,所以一般情况下我们不用特意去把用完的对象设置为null。那这种适合在什么时候用呢?比如在方法中我们有一个大对象,这个大对象占用内大内存,在大对象用完后,其后面还有很多耗时的业务代码,正常情况下这个大对象回收要等到耗时的业务代码执行完后才会释放内存,如果想提前释放,可以考虑把他的引用设置为null

     public static void main(String[] args) throws InterruptedException {
            Object bigObject = new Object();
         	//模拟使用bigObject
         	use(bigObject);
         	//用完大对象设置null,让其在此函数执行完之前就有机会被回收
         	bigObject = null;
         
            //模拟一些耗时代码
            xxxxxx
        }
    
  2. 栈帧中的局部变量表中的槽位是可以重用的,如果一个局部变量过了其作用域,那么在其作用域之后申明的新的局部变量就很有可能会复用过期局部变量的槽位,从而达到节省资源的目的。

    在第三次测试中我们用代码块包围了KB_512,来限制其作用于,我们发现也能达到在main方法执行前就释放内存的目的。原因就是KB_512这个变量的作用于限制在代码块,代码快执行完后,就用不上它了,那么它在栈帧中的局部变量那个地址是可被别的局部变量复用,即temp复用了。当temp把引用指向new byte[0];时,原来的new byte[1 * 1024 * 512];就没有变量去引用它了,所以它可以被回收。

3、参考

jvm内存结构

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: RLock lock = null意义是在声明一个RLock类型的变量lock,并将其初始化为null。这种方式可以避免在声明变量时自动给变量赋初始值,从而提高代码的可读性和可维护性。另外,在使用该变量前需要注意判断其是否为null,避免引发空指针异常。 ### 回答2: RLock lock = null意义是在代码中声明了一个可重入锁(ReentrantLock)对象 lock,并将其初始化为 null。可重入锁是一种支持线程重入的锁,即同一个线程可以多次获取该锁而不会造成死锁。 将 lock 初始化为 null 的目的是为了延迟创建锁对象,只有在需要使用锁的时候才创建它,避免在初始化阶段就进行锁的创建,提高代码的执行效率。 此外,将 lock 声明为 RLock 类型,可以让代码在后续的编写中更加清晰明了,让其他开发人员能够迅速理解该变量的作用和用途。 但需要注意的是,当在代码中使用 lock 对象时,应该先检查 lock 是否为 null,如果为 null,则需要进行锁对象的创建和初始化。这样的检查是为了防止在使用 lock 对象前未进行初始化而出现空指针异常。 总结来说,RLock lock = null意义是为了在代码中声明一个可重入锁对象并延迟创建,提高代码效率,并在使用该对象前进行 null 检查,避免空指针异常的发生。 ### 回答3: RLock lock = null意义是在声明一个名为lock的RLock类型的变量,并将其初始化为null。RLock(可重入锁)是Java中用于多线程同步的一种机制,它允许线程获取同一把锁多次并进行嵌套调用。 将lock初始化为null意义在于表明在声明这个变量时,我们还没有指定具体的可重入锁对象。在代码的后续部分,我们需要根据具体的需求和逻辑,选择合适的可重入锁对象来进行初始化操作。 这种声明的目的主要是为了提醒程序员在后续使用lock对象之前,必须为其赋予一个有效的RLock对象。否则,如果在给lock对象赋值之前就直接使用它,可能会导致空指针异常或其他错误。 当我们需要使用可重入锁时,可以根据具体的情况选择适当的实现方式,例如ReentrantLock类,它是Java标准库提供的一种实现。 总之,RLock lock = null意义是声明一个可重入锁对象的变量,并将其初始化为null,用来提醒在使用该对象之前必须为其赋值,防止出现异常或错误。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值