深入理解Java的4种引用类型

简介

首先大家应该都知道Java从1.2起提供了四种引用类型,他们分别是其引用(StrongReference),软引用(SoftReference),弱引用(WeakReference)以及PhantomReference(虚引用),他们被GC回收的可能性从大到小排列。如下图

可以看到Reference是继承自Object,而又有三个直接的子类,就是我们要介绍的几个类了。另外还有一个ReferenceQueue我想大家就不一定知道了

他是用来的保存即将释放的引用对象,具体用户下面有实例代码

下面就来分别介绍每个引用,并附上测试代码,方便大家的深入理解

StrongReference

就可以他的名字一样,任何时候GC是不能回收他的,哪怕内存不足时,系统会直接抛出异常OutOfMemoryError,也不会去回收,首先要说明的是java中默认就是强引用,比如如下代码:

Persion p = new Persion();

其中p就是一个强引用,任何时候都不会被gc回收

/**
 * 测试强类型,可以看到我们调用多次gc他还是没有回收
 */
private static void testStrongReference() {
    Person jack = new Person("Jack");
    System.gc();
    System.gc();
    System.out.println(jack);
}

SoftReference

软引用他的特点是当内存足够时不会回收这种引用类型的对象,只有当内存不够用时才会回收,这种特点很适合最一些缓存,比如android中图片的缓存

/**
 * 测试软类型引用
 */
private static void testSoftReference() {
    Person jack = new Person("Jack");
    SoftReference<Person> personSoftReference = new SoftReference<>(jack);
    System.out.println(personSoftReference.get());
    jack = null;
    System.gc();
    System.gc();
    System.out.println(personSoftReference.get());
}

输出:

Person{name='Jack'}
Person{name='Jack'}

可以看到虽然我们手动调用了GC但是引用类型没有被回收

WeakReference

虚引用的特点是只要GC一运行就会把给回收了,个人感觉没多大用处,因为只要GC一运行他就会被回收了‘

private static void testWeakReference() {
    Person jack = new Person("Jack");
    WeakReference<Person> personSoftReference = new WeakReference<Person>(jack);
    System.out.println(personSoftReference.get());
    System.gc();
    System.gc();
    System.out.println(personSoftReference.get());
}

我们运行代码发现对象还是没被回收,因为我们虽然调用了GC,这个方法只是通知他执行,但是什么执行还得看他心情。当然上面的代码有个很重要的原因是Jack对象他是强引用,所以在怎么调用GC他还是不会回收的,为啥呢?因为jack这个变量他还强引用着对象,我们给他置空,在调用GC他就回收了,如下代码:

private static void testWeakReference() {
    Person jack = new Person("Jack");
    WeakReference<Person> personSoftReference = new WeakReference<Person>(jack);
    System.out.println(personSoftReference.get());
    jack = null;
    System.gc();
    System.gc();
    System.out.println(personSoftReference.get());
}

输出:

Person{name='Jack'}
null

可以看到结果正如我们所料

PhantomReference

虚引用就相应没引用似得,主要以创建这种类型的引用,那么他所引用类型就回收了

private static void testPhantomReference() {
    Person jack = new Person("Jack");
    ReferenceQueue<Person> personReferenceQueue = new ReferenceQueue<>();
    PhantomReference<Person> personSoftReference = new PhantomReference<Person>(jack,personReferenceQueue);
    System.out.println(personSoftReference.get());
//        jack = null;
//        System.gc();
//        System.gc();
    System.out.println(personSoftReference.get());
}

输出:

null
null

可以看到只要一创建完就直接给回收了

ReferenceQueue

他是一个引用对列,可以监控被回收的Reference对象,具体的就是当一个引用对象他所引用的值被回收了,那么系统就会把这个引用添加到这个对列里面,如下代码:

private static void testReferenceQueue() {
    Person jack = new Person("Jack");
    ReferenceQueue<Person> personReferenceQueue = new ReferenceQueue<>();
    WeakReference<Person> personSoftReference = new WeakReference<Person>(jack, personReferenceQueue);
    jack = null;
    System.gc();
    System.out.println(personSoftReference.get());
}

当personSoftReference所引用的Jack对象回收了,系统就会把personSoftReference对象添加到personReferenceQueue里,具体有什么作用呢,想象下这种情景,一个hashMap他的key是一个byte[]数组,我们需要创建一万个这样的key将他添加到hashMap,这是如果机器的内存小那么他就可以内存不足,但是当我们用一个引用来包裹这个key,然后在添加到hashMap就不会了:

private static void testReferenceQueue1() {
    ReferenceQueue<Object> weakReferenceReferenceQueue = new ReferenceQueue<>();

    int M1 = 1024;
    Object o = new Object();
    Map<WeakReference<byte[]>, Object> map = new HashMap<>();

    //创建一个线程监听回收的对象
    new Thread(new Runnable() {
        @Override
        public void run() {
            try {
                int cnt = 0;
                WeakReference<byte[]> k;
                while((k = (WeakReference) weakReferenceReferenceQueue.remove()) != null) {
                    System.out.println((cnt++) + "recycle:" + k+","+k.get());
                    System.out.println("map.size:" + map.size());
                }

            } catch(InterruptedException e) {
                //结束循环

            }
        }
    }).start();

    for (int i = 0; i < 10000; i++) {
        byte[] bytes = new byte[M1];
        map.put(new WeakReference<>(bytes,weakReferenceReferenceQueue),o);
    }
    System.gc();
    System.out.println("map.size:" + map.size());
}

输出:

...
9997recycle:java.lang.ref.WeakReference@2a62b5bc,null
map.size:10000
9998recycle:java.lang.ref.WeakReference@27e47833,null
map.size:10000
9999recycle:java.lang.ref.WeakReference@18e8473e,null
map.size:10000

可以看到他回收了这么多对象,因为我们调用get()时返回的为null,但是问题来了,这时候map的size还是10000,这不是我们期望的,因为他所引用的对象都被回首了,那这个map里面的可以相当于失效了,我们希望的是把reference对象也回收了,意思是map的size变成真实可用对象的size

private static void testReferenceQueue2() {
    ReferenceQueue<Object> weakReferenceReferenceQueue = new ReferenceQueue<>();

    int M1 = 1024;
    Object o = new Object();
    Map<WeakReference<byte[]>, Object> map = new HashMap<>();

    //创建一个线程监听回收的对象
    new Thread(new Runnable() {
        @Override
        public void run() {
            try {
                int cnt = 0;
                WeakReference<byte[]> k;
                while ((k = (WeakReference) weakReferenceReferenceQueue.remove()) != null) {
                    map.remove(k); //在这里我们移除被回收对象的引用
                    System.out.println((cnt++) + "recycle:" + k);
                    System.out.println("map.size:" + map.size());
                }


            } catch (InterruptedException e) {
                //结束循环

            }
        }
    }).start();

    for (int i = 0; i < 10000; i++) {
        byte[] bytes = new byte[M1];
        map.put(new WeakReference<>(bytes, weakReferenceReferenceQueue), o);
    }
    System.gc();
    System.out.println("map.size:" + map.size());
}

输出:

9996recycle:java.lang.ref.WeakReference@1608bcbd
map.size:3
9997recycle:java.lang.ref.WeakReference@777c9dc9
map.size:2
9998recycle:java.lang.ref.WeakReference@535779e4
map.size:1
9999recycle:java.lang.ref.WeakReference@6f6745d6
map.size:0

可以看见被回收了的对象,我们将它的key也从map里面移除了

WeakHashMap

他的效果是和上面差不多,当一个key的引用对象被回收时他会自动的移除这个key

我们将下表是偶数的值装到了list,这样做是不让他回收这些对象

private static void testWeakHashMap() {
    ReferenceQueue<Object> weakReferenceReferenceQueue = new ReferenceQueue<>();

    int M1 = 1024;
    Object o = new Object();
    Map<byte[], Object> map = new WeakHashMap<>();

    ArrayList<byte[]> bytes1 = new ArrayList<>();
    for (int i = 0; i < 10000; i++) {
        byte[] bytes = new byte[M1];
        if (i%2==0){
            bytes1.add(bytes);
        }
        map.put(bytes, o);
        bytes  = null;
    }
    System.gc();
    System.out.println("map.size:" + map.size());
}

输出:

map.size:5000

可以看到正如我们上面说的,他会移除哪些被回收的key。

参考:http://www.iflym.com/index.php/java-programe/201407140001.html

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 160,999评论 4 368
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 68,102评论 1 302
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 110,709评论 0 250
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,439评论 0 217
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,846评论 3 294
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,881评论 1 224
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 32,062评论 2 317
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,783评论 0 205
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,517评论 1 248
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,762评论 2 253
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,241评论 1 264
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,568评论 3 260
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,236评论 3 241
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,145评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,941评论 0 201
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,965评论 2 283
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,802评论 2 275

推荐阅读更多精彩内容