JAVA引用类型

本文详细介绍了Java中四种不同级别的对象引用:强引用、软引用、弱引用和虚引用。每种引用都有不同的应用场景,例如使用软引用进行缓存管理,使用虚引用配合引用队列进行对象回收通知。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

     如果一个内存中的对象没有任何引用的话,就说明这个对象已经不再被使用了,从而可以成为被垃圾回收的候选。不过,由于垃圾回收器的运行时间不确定,所以可被垃圾回收的对象的实际被回收时间是不确定的。对于一个对象来说,只要有引用的存在,它就会一直存在于内存中。如果这样的对象越来越多,超出了JVM中的内存总数,JVM就会抛出OutOfMemory错误。


        虽然垃圾回收的具体运行是由JVM来控制的,但是开发人员仍然可以在一定程度上与垃圾回收器进行交互,其目的在于更好的帮助垃圾回收器管理好应用的内存。


        在Java中把对象的引用分为 4 种级别,从而使程序能更加灵活地控制对象的生命周期。这 4 种级别由高到低依次为:强引用、软引用、弱引用和虚引用。


        JAVA对象引用类层次结构图如下图所示:


  

图14-1  JAVA对象引用类层次结构图


        1)强引用类型


        在一般的Java程序中,见到最多的就是强引用(strong reference)。如我们最常见的代码


[代码]java代码:

1//变量定义
2String str = “Hello Wolrd”;
3        Object obj = new Object();
4        Date date = new Date(),


        在上面的代码中str、obj、date都是对象的强引用。对象的强引用可以在程序中到处传递。很多情况下,会同时有多个引用指向同一个对象。强引用的存在限制了对象在内存中的存活时间。假如对象A中包含了一个对象B的强引用,那么一般情况下,对象B的存活时间就不会短于对象A。如果对象A没有显式的把对象B的引用设为null的话,就只有当对象A被垃圾回收之后,对象B才不再有引用指向它,才可能获得被垃圾回收的机会。


        2)软引用类型


        软引用(soft reference)在强度上弱于强引用,通过SoftReference类来表示。它的作用是告诉垃圾回收器,程序中的哪些对象并不那么重要,当内存不足的时候是可以被暂时回收的。当JVM中的内存不足的时候,垃圾回收器会释放那些只被软引用所指向的对象。如果全部释放完这些对象之后,内存还不足,才会抛出Out Of Memory错误。


        软引用非常适合于创建缓存。当系统内存不足的时候,缓存中的内容是可以被释放的。比如考虑一个图像编辑器的程序,该程序会把图像文件的全部内容都读取到内存中,以方便进行处理,而用户也可以同时打开多个文件。当同时打开的文件过多的时候,就可能造成内存不足。如果使用软引用来指向图像文件内容的话,垃圾回收器就可以在必要的时候回收掉这些内存。


        从下面的Java代码中可以看到软引用类型的使用方法。


[代码]java代码:

01public class BitmapCache {
02        private String url;//图片URL
03        private SoftReference<Bitmap> softRef;// //软引用-只有当系统内存不足的时候才去释放
04    public BitmapCache (String url) {
05         this. url = url;
06         softRef = new SoftReference< Bitmap >(null);
07    }
08    private Bitmap loadRemoteBitmap() {
09        final DefaultHttpClient client = new DefaultHttpClient();
10        final HttpGet getRequest = new HttpGet(url);
11        HttpResponse response = client.execute(getRequest);
12        final int statusCode = response.getStatusLine().getStatusCode();
13        final HttpEntity entity = response.getEntity();
14        InputStream inputStream = entity.getContent();
15        final ByteArrayOutputStream dataStream = new ByteArrayOutputStream();
16        OutputStream outputStream = new BufferedOutputStream(dataStream, IO_BUFFER_SIZE);
17        copy(inputStream, outputStream);
18        outputStream.flush();
19        final byte[] data = dataStream.toByteArray();
20        final Bitmap bitmap = BitmapFactory.decodeByteArray(data, 0,  data.length);
21        return bitmap;
22    }
23    public Bitmap get Bitmap () {
24            Bitmap bitmap = softRef.get();
25        if (bitmap == null) {//系统内存不足的时,图片已经被释放需要重新加载网络图片
26            bitmap = loadRemoteBitmap ();
27                softRef = new SoftReference< Bitmap >( bitmap);
28        }
29        return bitmap;
30        }
31}


        在使用上面程序的时候,由于软引用所指向的对象可能被回收掉,在通过get方法来获取软引用所实际指向的对象的时候,总是要检查该对象是否还存活。


        3)弱引用类型


        弱引用(weak reference)在强度上弱于软引用,通过WeakReference类来表示。它的作用是引用一个对象,但是并不阻止该对象被回收。如果使用一个强引用的话,只要该引用存在,那么被引用的对象是不能被回收的,弱引用则没有这个问题。在垃圾回收器运行的时候,如果一个对象的所有引用都是弱引用的话,该对象会被回收。弱引用的作用在于解决强引用所带来的对象之间在存活时间上的耦合关系。


        弱引用最常见的用处是在集合类中,尤其在哈希表中。哈希表的接口允许使用任何Java对象作为键(Key)。当一个键值对(Key-Value)被放入到哈希表中之后,哈希表对象本身就有了对这些键和值对象的引用。如果这种引用是强引用的话,那么只要哈希表对象本身还存活,其中所包含的键和值对象是不会被回收的。如果某个存活时间很长的哈希表中包含的键值对很多,最终就有可能消耗掉JVM中全部的内存。对于这种情况的解决办法就是使用弱引用来引用这些对象,这样哈希表中的键和值对象都能被垃圾回收器及时回收。在Java中可以使用WeakHashMap类来满足这一常见需求。


        4)虚引用类型


        在介绍虚引用之前,先要了解一下Java提供的对象终止化机制(finalization)。


        在Object类里面有个finalize方法,其设计的初衷是在一个对象被真正回收之前,可以用来执行一些清理的工作。因为Java并没有提供类似C++的析构函数一样的机制,只是简单地通过 finalize方法来实现。但是问题在于垃圾回收器的运行时间是不固定的,所以这些清理工作的实际运行时间也是不能预知的。


        使用虚引用(phantom reference)可以解决这个问题。在创建虚引用PhantomReference的时候必须要指定一个引用队列。当一个对象的finalize方法已经被调用了之后,这个对象的虚引用会被加入到队列中。通过检查该队列里面的内容就知道一个对象是不是已经准备要被回收了。


        虚引用及其队列的使用情况并不多见,主要用来实现比较精细的内存使用控制,这对于移动设备来说是很有意义的。程序可以在确定一个对象要被回收之后,再申请内存创建新的对象。通过这种方式可以使得程序所消耗的内存维持在一个相对较低的数量。


        比如下面的Java代码给出了一个缓冲区的实现示例。


[代码]java代码:

01//虚引用实例代码
02public class PhantomBuffer {
03//缓冲区字节数组
04private byte[] data = new byte[0];
05//引用队列
06private ReferenceQueue<byte[]> queue = new ReferenceQueue<byte[]>();
07//虚引用
08private PhantomReference<byte[]> ref = new PhantomReference<byte[]>(data, queue);
09//生成指定大小的缓冲区
10public byte[] get(int size) {
11        //校验size是否合法
12        if (size <= 0) {
13            // size不合法,抛出异常
14            throw new IllegalArgumentException("Wrong buffer size");
15        }
16        //检查当前缓冲区是否能满足申请的缓冲区大小
17        if (data.length < size) {
18            data = null;
19           //强制运行垃圾回收器
20            System.gc();
21             try {
22              //该方法会阻塞直到队列非空
23                queue.remove();
24              //虚引用不会自动清空,要手动运行
25                ref.clear();
26                ref = null;
27              //创建新的缓冲区
28                data = new byte[size];
29              //创建虚引用,并加入到应用队列
30                ref = new PhantomReference<byte[]>(data, queue);
31            catch (InterruptedException e) {
32                e.printStackTrace();
33            }
34       }
35       return data;
36    }
37}


        在上面的代码中,每次申请新的缓冲区的时候,都首先确保之前的缓冲区的字节数组已经被成功回收。引用队列的remove方法会阻塞直到新的虚引用被加入到队列中。需要注意的是,这种做法会导致垃圾回收器被运行的次数过多,可能会造成程序的吞吐量过低。


        5)引用队列


        在有些情况下,程序会需要在一个对象的可达到性发生变化的时候得到通知。比如某个对象的强引用都已经不存在了,只剩下软引用或是弱引用。但是还需要对引用本身做一些的处理。典型的情景是在哈希表中,引用对象是作为WeakHashMap中的键对象的,当其引用的实际对象被垃圾回收之后,就需要把该键值对从哈希表中删除。有了引用队列(ReferenceQueue),就可以方便的获取到这些弱引用对象,将它们从表中删除。在软引用和弱引用对象被添加到队列之前,其对实际对象的引用会被自动清空。通过引用队列的poll/remove方法就可以分别以非阻塞和阻塞的方式获取队列中的引用对象。

转载于:https://my.oschina.net/u/1989867/blog/382413

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值