[Java开发之路](25)引用类型

简介: 版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/SunnyYoona/article/details/51055815 1. 强引用 Java中的引用,类似于C++的指针。
版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/SunnyYoona/article/details/51055815

1. 强引用

Java中的引用,类似于C++的指针。通过引用,可以对堆中的对象进行操作。在某个函数中,当创建了一个对象,该对象被分配在堆中,通过这个对象的引用才能对这个对象进行操作。

  
  
  1. StringBuffer str = new StringBuffer("hello world");

假设以上代码是在函数体内运行的,那么局部变量str将被分配在栈上,而对象StringBuffer实例,被分配在堆上。局部变量str指向StringBuffer实例所在的堆空间,通过str可以操作该实例,那么str就是StringBuffer的引用。


此时,运行一个赋值语句:

  
  
  1. StringBuffer str1 = str;

那么,str所指向的对象也将被str1所指向,同时在局部栈空间上会分配空间存放str1变量。此时,该StringBuffer实例就有两个引用。对引用的"=="操作用于表示两个操作数所指向的堆空间地址是否相同,不表示两个操作数所指向的对象是否相等。


强引用特点:

  • 强引用可以直接访问目标对象。

  • 强引用所指向的对象在任何时候都不会被系统回收。JVM宁愿抛出OOM异常,也不会回收强引用所指向的对象。

  • 强引用可能导致内存泄露。


2. 软引用

软引用是除了强引用外,最强的引用类型。可以通过java.lang.ref.SoftReference使用软引用。一个持有软引用的对象,不会被JVM很快回收,JVM会根据当前堆的使用情况来判断何时回收。当堆的使用率临近阈值时,才会回收软引用的对象。

   
   
  1. package com.qunar.base;
  2. import java.lang.ref.Reference;
  3. import java.lang.ref.ReferenceQueue;
  4. import java.lang.ref.SoftReference;
  5. /**
  6. * Created by xiaosi on 16-3-24.
  7. */
  8. public class ReferenceDemo {
  9.    // 创建引用队列
  10.    private ReferenceQueue<MyObject> referenceQueue = new ReferenceQueue<>();
  11.    public class MyObject {
  12.        @Override
  13.        protected void finalize() throws Throwable {
  14.            super.finalize();
  15.            // 被回收时输出
  16.            System.out.println("MyObject is finalize called");
  17.        }
  18.        @Override
  19.        public String toString() {
  20.            return " I am  MyObject";
  21.        }
  22.    }
  23.    public class CheckRefQueue implements Runnable {
  24.        Reference<MyObject> obj = null;
  25.        @Override
  26.        public void run() {
  27.            try {
  28.                // 如果对象被回收则进入引用队列
  29.                obj = (Reference<MyObject>) referenceQueue.remove();
  30.            } catch (InterruptedException e) {
  31.                e.printStackTrace();
  32.            }
  33.            if (obj != null) {
  34.                System.out.println("Object for SoftReference is " + obj.get());
  35.            }
  36.        }
  37.    }
  38.    public void test() {
  39.        // 创建强引用
  40.        MyObject myObject = new MyObject();
  41.        // 构造myObject对象的软引用 注册到 引用队列
  42.        SoftReference<MyObject> softReference = new SoftReference<>(myObject, referenceQueue);
  43.        CheckRefQueue checkRefQueue = new CheckRefQueue();
  44.        Thread thread = new Thread(checkRefQueue);
  45.        thread.start();
  46.        // 删除强引用 对myObject对象的引用只剩下软引用
  47.        myObject = null;
  48.        System.gc();
  49.        System.out.println("After GC: Soft Get = " + softReference.get());
  50.        System.out.println("分配大块内存");
  51.        // 分配一块强大的内存,强迫GC
  52.        byte[] b = new byte[5*1024*963];
  53.        System.out.println("After new byte[]:Soft Get = "+softReference.get());
  54.        System.gc();
  55.    }
  56.    public static void main(String[] args) {
  57.        ReferenceDemo referenceDemo = new ReferenceDemo();
  58.        referenceDemo.test();
  59.    }
  60. }

首先构造MyObject对象,并将其赋值给myObject变量,构成强引用。然后使用SoftReference构造这个MyObject对象的软引用softReference,并注册到referenceQueue队列中。当softReference被回收时,会被加入referenceQueue队列。设置myObject=null,删除这个强引用,因此,系统内对MyObject对象的引用只剩下软引用。此时,显示调用GC,通过软引用的get()方法,取得MyObject对象实例的强引用,发现对象被未回收。这说明GC在内存充足的情况下,不会回收软引用对象。


接着,请求一块大的堆空间new byte[4*1024*925],这个操作会使操作系统内存使用紧张,从而产生新一轮的GC。这次GC后,softReference.get()不再返回MyObject对象,而是返回null,说明在系统内存紧张的情况下,软引用被回收。软引用被回收时,会被加入到注册的引用队列中。

备注:

JVM参数:-Xmx5M

运行结果:

/opt/jdk1.7.0_40/bin/java -Xmx5M ...

After GC: Soft Get =  I am  MyObject

分配大块内存

After new byte[]:Soft Get = null

MyObject is finalize called

Object for SoftReference is null


3. 弱引用

弱引用是一种比软引用较弱的引用类型。在系统GC时,只要发现弱引用,不管系统堆空间是否足够,都会将对象进行回收。但是,由于垃圾回收器的线程通常优先级很低,因此,并一不定能很快的发现持有弱引用的对象。这种情况下,弱引用对象可以存在较长的一段时间。一旦一个弱引用对象被垃圾回收器回收,便会加入到一个注册引用队列中。

   
   
  1. package com.qunar.base;
  2. import java.lang.ref.Reference;
  3. import java.lang.ref.ReferenceQueue;
  4. import java.lang.ref.SoftReference;
  5. import java.lang.ref.WeakReference;
  6. /**
  7. * Created by xiaosi on 16-3-24.
  8. */
  9. public class WeakReferenceDemo {
  10.    // 创建引用队列
  11.    private ReferenceQueue<MyObject> referenceQueue = new ReferenceQueue<>();
  12.    public class MyObject {
  13.        @Override
  14.        protected void finalize() throws Throwable {
  15.            super.finalize();
  16.            // 被回收时输出
  17.            System.out.println("MyObject is finalize called");
  18.        }
  19.        @Override
  20.        public String toString() {
  21.            return " I am  MyObject";
  22.        }
  23.    }
  24.    public class CheckRefQueue implements Runnable {
  25.        Reference<MyObject> obj = null;
  26.        @Override
  27.        public void run() {
  28.            try {
  29.                // 如果对象被回收则进入引用队列
  30.                obj = (Reference<MyObject>) referenceQueue.remove();
  31.            } catch (InterruptedException e) {
  32.                e.printStackTrace();
  33.            }
  34.            if (obj != null) {
  35.                System.out.println("Object for SoftReference is " + obj.get());
  36.            }
  37.        }
  38.    }
  39.    public void test() {
  40.        // 创建强引用
  41.        MyObject myObject = new MyObject();
  42.        // 构造myObject对象的弱引用 注册到 引用队列
  43.        WeakReference<MyObject> weakReference = new WeakReference<>(myObject, referenceQueue);
  44.        CheckRefQueue checkRefQueue = new CheckRefQueue();
  45.        Thread thread = new Thread(checkRefQueue);
  46.        thread.start();
  47.        // 删除强引用 对myObject对象的引用只剩下弱引用
  48.        myObject = null;
  49.        System.out.println("Before GC: Weak Get = " + weakReference.get());
  50.        System.gc();
  51.        System.out.println("After GC: Weak Get = " + weakReference.get());
  52.    }
  53.    public static void main(String[] args) {
  54.        WeakReferenceDemo referenceDemo = new WeakReferenceDemo();
  55.        referenceDemo.test();
  56.    }
  57. }

运行结果:

Before GC: Weak Get =  I am  MyObject

After GC: Weak Get = null

MyObject is finalize called

Object for SoftReference is null


在GC之前,弱引用对象并未被垃圾回收器发现,因此通过weakRef.get()方法可以取得对应的强引用。但是只要进行垃圾回收,弱引用对象一旦被发现,便会立即被回收,并加入注册引用队列中。此时,再次通过weakRef.get()方法取得强引用就会失败。


备注:

软引用,弱引用都非常适合来保存那些可有可无的缓存数据。如果这样做,当系统内存不足时,这些缓存数据会被回收,不会导致内存溢出。而当内存资源充足时,这些缓存数据又可以存在相当长的时间。


4. 虚引用

虚引用是所有引用类型中最弱的一个。一个持有虚引用的对象,和没有引用几乎是一样的,随时都可能被垃圾回收器回收。当试图通过虚引用的get()方法取得强引用时,总是会失败。并且,虚引用必须和引用队列一起使用,它的作用在于跟踪垃圾回收过程。

当垃圾回收器准备回收一个对象时,如果发现它还有虚引用,就会在垃圾回收后,销毁这个对象,奖这个虚引用加入引用队列。

  
  
  1. package com.qunar.base;
  2. import java.lang.ref.PhantomReference;
  3. import java.lang.ref.Reference;
  4. import java.lang.ref.ReferenceQueue;
  5. import java.lang.ref.WeakReference;
  6. /**
  7. * Created by xiaosi on 16-3-24.
  8. */
  9. public class PhantomReferenceDemo {
  10.    // 创建引用队列
  11.    private ReferenceQueue<MyObject> referenceQueue = new ReferenceQueue<>();
  12.    public class MyObject {
  13.        @Override
  14.        protected void finalize() throws Throwable {
  15.            super.finalize();
  16.            // 被回收时输出
  17.            System.out.println("MyObject is finalize called");
  18.        }
  19.        @Override
  20.        public String toString() {
  21.            return " I am  MyObject";
  22.        }
  23.    }
  24.    public class CheckRefQueue implements Runnable {
  25.        Reference<MyObject> obj = null;
  26.        @Override
  27.        public void run() {
  28.            try {
  29.                // 如果对象被回收则进入引用队列
  30.                obj = (Reference<MyObject>) referenceQueue.remove();
  31.                // 等待 直到取得虚引用对象
  32.                System.out.println("Object for PhantomReference is " + obj.get());
  33.                System.exit(0);
  34.            } catch (InterruptedException e) {
  35.                e.printStackTrace();
  36.            }
  37.            if (obj != null) {
  38.                System.out.println("Object for SoftReference is " + obj.get());
  39.            }
  40.        }
  41.    }
  42.    public void test() throws InterruptedException {
  43.        // 创建强引用
  44.        MyObject myObject = new MyObject();
  45.        // 构造myObject对象的虚引用 注册到 引用队列
  46.        PhantomReference<MyObject> phantomReference = new PhantomReference<>(myObject, referenceQueue);
  47.        System.out.println("phantomReference Get : " + phantomReference.get());
  48.        CheckRefQueue checkRefQueue = new CheckRefQueue();
  49.        Thread thread = new Thread(checkRefQueue);
  50.        thread.start();
  51.        // 删除强引用 对myObject对象的引用只剩下虚引用
  52.        myObject = null;
  53.        Thread.sleep(1000);
  54.        int i = 1;
  55.        while(true){
  56.            System.out.println("第" + i + "次GC");
  57.            System.gc();
  58.            Thread.sleep(1000);
  59.            i++;
  60.        }//while
  61.    }
  62.    public static void main(String[] args) throws InterruptedException {
  63.        PhantomReferenceDemo referenceDemo = new PhantomReferenceDemo();
  64.        referenceDemo.test();
  65.    }
  66. }

运行结果:

phantomReference Get : null

第1次GC

MyObject is finalize called

第2次GC

Object for PhantomReference is null


从这个输出结果中可以看出,对虚引用的get()操作,总是返回null,即便强引用还存在时,也不例外。因为虚引用的get()实现:

  
  
  1.    public T get() {
  2.        return null;
  3.    }

在第一次GC时,系统找到了垃圾对象,并调用其finalize()方法回收内存,但没有立即加入到回收队列中。第二次GC时,该对象真正的被GC清除,此时,加入虚引用队列。


虚引用最大作用在于跟踪对象回收,清理被销毁对象的相关资源。通常,当对象不被使用时,重载该类的finalize()方法可以回收该对象的资源。但是,如果finalize()方法使用不慎,可能导致该对象复活

下面看一个错误的finalize()实现的例子,这个实现导致内存溢出,或者对象永远无法被回收

  
  
  1. package com.qunar.base;
  2. /**
  3. * Created by xiaosi on 16-3-24.
  4. */
  5. public class PhantomObject {
  6.    public static PhantomObject phantomObject;
  7.    @Override
  8.    protected void finalize() throws Throwable {
  9.        super.finalize();
  10.        // 被回收时输出
  11.        System.out.println("MyObject is finalize called");
  12.        // 在finalize()中拯救了将要被回收的对象
  13.        phantomObject = this;
  14.    }
  15.    @Override
  16.    public String toString() {
  17.        return " I am  MyObject";
  18.    }
  19.    public static void main(String[] args) throws InterruptedException {
  20.        phantomObject = new PhantomObject();
  21.        // 删除对象
  22.        phantomObject = null;
  23.        // 第一次GC
  24.        System.gc();
  25.        Thread.sleep(1000);
  26.        if(phantomObject == null){
  27.            System.out.println("phantomObject is null");
  28.        }//if
  29.        else{
  30.            System.out.println("phantomObject 可用");
  31.        }
  32.        System.out.println("第二次GC");
  33.        System.gc();
  34.        Thread.sleep(1000);
  35.        if(phantomObject == null){
  36.            System.out.println("phantomObject is null");
  37.        }//if
  38.        else{
  39.            System.out.println("phantomObject 可用");
  40.        }
  41.        System.out.println("第三次GC");
  42.        System.gc();
  43.        Thread.sleep(1000);
  44.        if(phantomObject == null){
  45.            System.out.println("phantomObject is null");
  46.        }//if
  47.        else{
  48.            System.out.println("phantomObject 可用");
  49.        }
  50.    }
  51. }

在上面代码中,先将对象phantomObject设置为null,告知GC这是一个需要清理的对象。然后,进行一次显示的GC,GC过后发现,虽然该对象的finalize()方法被调用,但是对象依然存在。随后,进行第二次GC,由于在GC之前没有清除对象的强引用,所以phantomObject依然没有被回收。

运行结果:

MyObject is finalize called

phantomObject 可用

第二次GC

phantomObject 可用

第三次GC

phantomObject 可用


可见,虽然finalize()被调用,但是phantomObject始终都没有被回收。如果要强制回收phantomObject,需要在第二次G前,使用phantomObject=null,去除该对象的强引用。由于finalize()只会被调用一次,因此在第二次回收时,对象就没有机会复活了。

运行结果:

MyObject is finalize called

phantomObject 可用

第二次GC

phantomObject is null

第三次GC

phantomObject is null


由于在finalize()中存在让回收对象复活的可能性,因此,在一个复杂的应用系统中,一旦finalize()方法实现有问题,就很容易造成内存泄露。而使用虚引用来清理相关资源则不会有类似的问题,因为虚引用队列中对象,事实上已经完成了对象的回收工作,是不可能再度复活该对象的。











目录
相关文章
|
11天前
|
Java API Maven
如何使用Java开发抖音API接口?
在数字化时代,社交媒体平台如抖音成为生活的重要部分。本文详细介绍了如何用Java开发抖音API接口,从创建开发者账号、申请API权限、准备开发环境,到编写代码、测试运行及注意事项,全面覆盖了整个开发流程。
54 10
|
17天前
|
SQL 安全 Java
安全问题已经成为软件开发中不可忽视的重要议题。对于使用Java语言开发的应用程序来说,安全性更是至关重要
在当今网络环境下,Java应用的安全性至关重要。本文深入探讨了Java安全编程的最佳实践,包括代码审查、输入验证、输出编码、访问控制和加密技术等,帮助开发者构建安全可靠的应用。通过掌握相关技术和工具,开发者可以有效防范安全威胁,确保应用的安全性。
35 4
|
18天前
|
缓存 监控 Java
如何运用JAVA开发API接口?
本文详细介绍了如何使用Java开发API接口,涵盖创建、实现、测试和部署接口的关键步骤。同时,讨论了接口的安全性设计和设计原则,帮助开发者构建高效、安全、易于维护的API接口。
48 4
|
29天前
|
开发框架 JavaScript 前端开发
HarmonyOS UI开发:掌握ArkUI(包括Java UI和JS UI)进行界面开发
【10月更文挑战第22天】随着科技发展,操作系统呈现多元化趋势。华为推出的HarmonyOS以其全场景、多设备特性备受关注。本文介绍HarmonyOS的UI开发框架ArkUI,探讨Java UI和JS UI两种开发方式。Java UI适合复杂界面开发,性能较高;JS UI适合快速开发简单界面,跨平台性好。掌握ArkUI可高效打造符合用户需求的界面。
88 8
|
24天前
|
SQL Java 程序员
倍增 Java 程序员的开发效率
应用计算困境:Java 作为主流开发语言,在数据处理方面存在复杂度高的问题,而 SQL 虽然简洁但受限于数据库架构。SPL(Structured Process Language)是一种纯 Java 开发的数据处理语言,结合了 Java 的架构灵活性和 SQL 的简洁性。SPL 提供简洁的语法、完善的计算能力、高效的 IDE、大数据支持、与 Java 应用无缝集成以及开放性和热切换特性,能够大幅提升开发效率和性能。
|
24天前
|
存储 Java 关系型数据库
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践,包括连接创建、分配、复用和释放等操作,并通过电商应用实例展示了如何选择合适的连接池库(如HikariCP)和配置参数,实现高效、稳定的数据库连接管理。
45 2
|
24天前
|
监控 Java 数据库连接
在Java开发中,数据库连接管理是关键问题之一
在Java开发中,数据库连接管理是关键问题之一。本文介绍了连接池技术如何通过预创建和管理数据库连接,提高数据库操作的性能和稳定性,减少资源消耗,并简化连接管理。通过示例代码展示了HikariCP连接池的实际应用。
19 1
|
18天前
|
安全 Java 测试技术
Java开发必读,谈谈对Spring IOC与AOP的理解
Spring的IOC和AOP机制通过依赖注入和横切关注点的分离,大大提高了代码的模块化和可维护性。IOC使得对象的创建和管理变得灵活可控,降低了对象之间的耦合度;AOP则通过动态代理机制实现了横切关注点的集中管理,减少了重复代码。理解和掌握这两个核心概念,是高效使用Spring框架的关键。希望本文对你深入理解Spring的IOC和AOP有所帮助。
31 0
WK
|
24天前
|
开发框架 移动开发 Java
C++和Java哪个更适合开发移动应用
本文对比了C++和Java在移动应用开发中的优劣,从市场需求、学习难度、开发效率、跨平台性和应用领域等方面进行了详细分析。Java在Android开发中占据优势,而C++则适合对性能要求较高的场景。选择应根据具体需求和个人偏好综合考虑。
WK
42 0
WK
|
24天前
|
安全 Java 编译器
C++和Java哪个更适合开发web网站
在Web开发领域,C++和Java各具优势。C++以其高性能、低级控制和跨平台性著称,适用于需要高吞吐量和低延迟的场景,如实时交易系统和在线游戏服务器。Java则凭借其跨平台性、丰富的生态系统和强大的安全性,广泛应用于企业级Web开发,如企业管理系统和电子商务平台。选择时需根据项目需求和技术储备综合考虑。
WK
39 0
下一篇
无影云桌面