java强、弱、软、虚引用

2018-03-14  本文已影响0人  肚皮怪_Sun
public class QuoteTest {
    public static void main(String[] args) {
    /*================================*/
    ///////////////////////////////////////////////////////////////////////////
    // 如果一个对象具有强引用,那就类似于必不可少的生活用品,垃圾回收器绝不会回收它,当内存空间不足,
    //Java虚拟机宁愿抛出outofMermoryError错误,使程序异常终止,也不会靠随意回收具有强引用的对象来解决内存不足问题
    //
    ///////////////////////////////////////////////////////////////////////////

    /*在heap堆中创建新的Object对象通过obj引用这个对象,第二句是通过obj建立o1到new Object()这个heap堆中的对象的引用
    *这两个引用都是强引用,只要存在heap中对象的引用,gc就不会收集该对象
    */
    Object obj = new Object();

    Object o1 = obj;

    obj = null;
    o1 = null;

    //对象属于那种可及的对象,由它的最强的引用决定
    String abc = new String("abc");

    SoftReference<String> abcSoftRef = new SoftReference<String>(abc);//软引用

    WeakReference<String> abcWeakRef = new WeakReference<String>(abc);//弱引用

    abc = null;//执行这行代码的时候对象不再是强可及,变成软可及的。

    abcSoftRef = null;//对象变成弱可及

//  weakTemp();

    test();

    }
    public static void softTemp() {
       A a = new A();
       SoftReference<A> aSoftRef = new SoftReference<>(a);

      if (aSoftRef != null) {
          a = aSoftRef.get();
       } else {
         a = new A();
        aSoftRef = new SoftReference<A>(a);
      }
    }
    ///////////////////////////////////////////////////////////////////////////
    // 弱引用于软引用的区别在于,只具有弱引用的对象拥有跟短暂的生命周. 期,在垃圾回收器线程扫描
    // ,一旦发现持有了弱引用的对象,不管当前内存空间足够与否,他都会回收它的内存
    // 弱引用可以和一个引用队列联合使用,如果弱引用所引用的对象被垃圾回收,Java虚拟机就会吧
    // 弱引用加入到与之关联的引用队列当中
    ///////////////////////////////////////////////////////////////////////////
   public static void weakTemp() {
      String abc = new String("abc");
      WeakReference<String> abcWeakRef = new WeakReference<>(abc);
      abc = null;

      System.out.println("before gc:" + abcWeakRef.get());
      System.gc();

      System.out.println("after gc :" + abcWeakRef.get());
    }
      public static boolean isRunning = true;


      ///////////////////////////////////////////////////////////////////////////
      // 虚引用 必须和引用队列关联使用
      ///////////////////////////////////////////////////////////////////////////
  public static void test() {
      String abc = new String("abc");
       System.out.println(abc.getClass() + " @" + abc.hashCode());

      final ReferenceQueue<String> referenceQueue = new ReferenceQueue<>();
      new Thread() {
          public void run() {
            while (isRunning) {
                Object poll = referenceQueue.poll();
                if (poll != null) {
                    try {
                        Field rereferent = Reference.class.getDeclaredField("referent");
                        rereferent.setAccessible(true);
                        Object result = rereferent.get(poll);
                        System.out.println("gc will  collect:" + result.getClass() + " @" + result.hashCode());
                    } catch (NoSuchFieldException e) {
                        e.printStackTrace();
                    } catch (IllegalAccessException e) {
                        e.printStackTrace();
                    }

                }
            }
        }
    }.start();

    PhantomReference<String> abcPhRef = new PhantomReference<>(abc, referenceQueue);
    abc = null;
    try {

        Thread.currentThread().sleep(3000);
        System.gc();
        Thread.currentThread().sleep(3000);
        isRunning = false;
    } catch (InterruptedException e) {
        e.printStackTrace();
    }

  }


   public static class A {}
}
上一篇 下一篇

猜你喜欢

热点阅读