Java、Android引用类型

server/2024/11/25 22:01:39/

Java/Android中有四种引用类型,分别是:
Strong reference - 强引用
Soft Reference - 软引用
Weak Reference - 弱引用
Phantom Reference - 虚引用

不同的引用类型有着不同的特性,同时也对应着不同的使用场景。

Strong reference - 强引用

实际编码中最常见的一种引用类型。常见形式如:A a = new A();等。强引用本身存储在栈内存中,其存储指向对内存中对象的地址。一般情况下,当对内存中的对象不再有任何强引用指向它时,垃圾回收机器开始考虑可能要对此内存进行的垃圾回收。如当进行编码:a = null,此时,刚刚在堆中分配地址并新建的a对象没有其他的任何引用,当系统进行垃圾回收时,堆内存将被垃圾回收。

SoftReference、WeakReference、PhantomReference都是类java.lang.ref.Reference的子类。Reference作为抽象基类,定义了其子类对象的基本操作。Reference子类都具有如下特点:

  • Reference子类不能无参化直接创建,必须至少以强引用对象为构造参数,创建各自的子类对象;
  • 因为1中以强引用对象为构造参数创建对象,因此,使得原本强引用所指向的堆内存中的对象将不再只与强引用本身直接关联,与Reference的子类对象的引用也有一定联系。且此种联系将可能影响到对象的垃圾回收。

根据不同的子类对象对其指示对象(强引用所指向的堆内存中的对象)的垃圾回收不同的影响特点,分别形成了三个子类,即SoftReference、WeakReference和PhantomReference

Soft Reference - 软引用

软引用的一般使用形式如下:
A a = new A();
SoftReference srA = new SoftReference(a);
通过对象的强引用为参数,创建了一个SoftReference对象,并使栈内存中的wrA指向此对象。
此时,进行如下编码:a = null,对于原本a所指向的A对象的垃圾回收有什么影响呢?
先直接看一下下面一段程序的输出结果:

public class ReferenceTest {public static void main(String[] args) {A a = new A();SoftReference<A> srA = new SoftReference<A>(a);a = null;if (srA.get() == null) {System.out.println("a对象进入垃圾回收流程");} else {System.out.println("a对象尚未被回收" + srA.get());}// 垃圾回收System.gc();if (srA.get() == null) {System.out.println("a对象进入垃圾回收流程");} else {System.out.println("a对象尚未被回收" + srA.get());}}
}class A {}
输出结果为:
a对象尚未被回收A@4807ccf6
a对象尚未被回收A@4807ccf6

当 a = null后,堆内存中的A对象将不再有任何的强引用指向它,但此时尚存在srA引用的对象指向A对象。当第一次调用srA.get()方法返回此指示对象时,由于垃圾回收器很有可能尚未进行垃圾回收,此时get()是有结果的,这个很好理解。当程序执行System.gc();强制垃圾回收后,通过srA.get(),发现依然可以得到所指示的A对象,说明A对象并未被垃圾回收。那么,软引用所指示的对象什么时候才开始被垃圾回收呢?需要满足如下两个条件:
1.当其指示的对象没有任何强引用对象指向它;
2.当虚拟机内存不足时。
因此,SoftReference变相的延长了其指示对象占据堆内存的时间,直到虚拟机内存不足时垃圾回收器才回收此堆内存空间。

Weak Reference - 弱引用

同样的,软引用的一般使用形式如下:
A a = new A();
WeakReference wrA = new WeakReference(a);
当没有任何强引用指向此对象时, 其垃圾回收又具有什么特性呢?

public class ReferenceTest {public static void main(String[] args) {A a = new A();WeakReference<A> wrA = new WeakReference<A>(a);a = null;if (wrA.get() == null) {System.out.println("a对象进入垃圾回收流程");} else {System.out.println("a对象尚未被回收" + wrA.get());}// 垃圾回收System.gc();if (wrA.get() == null) {System.out.println("a对象进入垃圾回收流程");} else {System.out.println("a对象尚未被回收" + wrA.get());}}
}class A {}
输出结果为:
a对象尚未被回收A@52e5376a
a对象进入垃圾回收流程

输出的第一条结果解释同上。当进行垃圾回收后,wrA.get()将返回null,表明其指示对象进入到了垃圾回收过程中。因此,对弱引用特点总结为:
WeakReference不改变原有强引用对象的垃圾回收时机,一旦其指示对象没有任何强引用对象时,此对象即进入正常的垃圾回收流程。
那么,依据此特点,很可能有疑问:WeakReference存在又有什么意义呢?
其主要使用场景见于:当前已有强引用指向强引用对象,此时由于业务需要,需要增加对此对象的引用,同时又不希望改变此引用的垃圾回收时机,此时WeakReference正好符合需求,常见于一些与生命周期的场景中。

ReferenceQueue

对于SoftReference和WeakReference,还有一个构造器参数为ReferenceQueue,当SoftReference或WeakReference所指示的对象确实被垃圾回收后,其引用将被放置于ReferenceQueue中。注意上文中,当SoftReference或WeakReference的get()方法返回null时,仅是表明其指示的对象已经进入垃圾回收流程,此时对象不一定已经被垃圾回收。而只有确认被垃圾回收后,其引用才会被放置于ReferenceQueue中。

public static void main(String[] args) {A a = new A();ReferenceQueue referenceQueue = new ReferenceQueue();WeakReference<Object> weakReference = new WeakReference<Object>(a, referenceQueue);a = null;if(weakReference.get() == null){System.out.println("a对象进入垃圾回收流程");}else {System.out.println("a对象尚未被回收" + weakReference.get());}System.out.println("rq item:" + referenceQueue.poll());System.gc();if(weakReference.get() == null){System.out.println("a对象进入垃圾回收流程");}else {System.out.println("a对象尚未被回收" + weakReference.get());}System.out.println("rq item:" + referenceQueue.poll());}public static class A{@Overrideprotected void finalize() throws Throwable {System.out.println("A finalize");super.finalize();}}
输出结果为:
a对象尚未被回收@4554617c
rq item:null
a对象进入垃圾回收流程
rq item:null
A finalize

当调用System.gc()后,输出‘a对象进入垃圾回收流程’说明a已经开始进入了垃圾回收流程,但是还未被真正回收,因为Java对象被回收时会调用期finalize方法;此时我们立刻调用referenceQueue.poll()得到的是空,这样验证了对象只有gc已回收后WeakReference或SoftReference才会被加入ReferenceQueue。

现在我们再gc后,进行一段时间的延时后再调用referenceQueue.poll():
 public static void main(String[] args) {A a = new A();ReferenceQueue referenceQueue = new ReferenceQueue();WeakReference<Object> weakReference = new WeakReference<Object>(a, referenceQueue);a = null;if(weakReference.get() == null){System.out.println("a对象进入垃圾回收流程");}else {System.out.println("a对象尚未被回收" + weakReference.get());}System.out.println("rq item:" + referenceQueue.poll());System.gc();if(weakReference.get() == null){System.out.println("a对象进入垃圾回收流程");}else {System.out.println("a对象尚未被回收" + weakReference.get());}try {Thread.sleep(2000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println("rq item:" + referenceQueue.poll());}public static class A{@Overrideprotected void finalize() throws Throwable {System.out.println("A finalize");super.finalize();}}
输出结果
a对象尚未被回收@4554617c
rq item:null
a对象进入垃圾回收流程
A finalize
rq item:@74a14482

此时referenceQueue中存在了WeakReference,实验的结果也就不言而喻了。

PhantomReference虚引用

与SoftReference或WeakReference相比,PhantomReference主要差别体现在如下几点:

  • PhantomReference只有一个构造函数PhantomReference(T referent, ReferenceQueue<? super T> q),因此,PhantomReference使用必须结合ReferenceQueue;
  • 不管有无强引用指向PhantomReference的指示对象,PhantomReference的get()方法返回结果都是null。
public class ReferenceTest {public static void main(String[] args) {A a = new A();ReferenceQueue<A> rq = new ReferenceQueue<A>();PhantomReference<A> prA = new PhantomReference<A>(a, rq);System.out.println("prA.get():" + prA.get());a = null;System.gc();try {Thread.sleep(2000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println("rq item:" + rq.poll());}
}class A {}
输出结果为:
prA.get():null
rq item:java.lang.ref.PhantomReference@1da12fc0

代码中的Thread.sleep(1);作用与上例中相同,都是确保垃圾回收线程能够执行。否则,进进入垃圾回收流程而没有真正被垃圾回收的指示对象的虚引用是不会被加入到PhantomReference中的。

与WeakReference相同,PhantomReference并不会改变其指示对象的垃圾回收时机。ReferenceQueue的作用主要是用于监听Java对象是否已经被垃圾回收。

http://www.ppmy.cn/server/144905.html

相关文章

【Python】selenium获取定位元素大小、电脑屏幕的像素、屏幕尺寸信息、以及网页尺寸的方法

我们在使用selenium 进行自动化测试的时候&#xff0c;有时候会使用获取网页的大小做一些计算&#xff0c;例如测试进行滑动验证码时&#xff0c;需要计算滑块在屏幕的相对大小&#xff0c;就要用到以上这些部分的内容。 1.1 获取某个的位置信息&#xff1a; button_ driver…

【Ubuntu24.04】服务部署(虚拟机)

目录 0 背景1 安装虚拟机1.1 下载虚拟机软件1.2 安装虚拟机软件1.2 安装虚拟电脑 2 配置虚拟机2.1 配置虚拟机网络及运行初始化脚本2.2 配置服务运行环境2.2.1 安装并配置JDK172.2.2 安装并配置MySQL8.42.2.3 安装并配置Redis 3 部署服务4 总结 0 背景 你的服务部署在了你的计算…

AI模型---安装cuda与cuDNN

1.安装cuda 先打开cmd 输入nvidia-smi 查看显卡支持cuda对应的版本&#xff1a; 然后去英伟达官网下载cuda&#xff08;外网多刷几次&#xff09; https://developer.nvidia.com/cuda-toolkit-archive 注意对应版本 安装过程中如果显示如下图&#xff1a; 请安装visual Stu…

鸿蒙开发Hvigor插件动态生成代码

Hvigor允许开发者实现自己的插件&#xff0c;开发者可以定义自己的构建逻辑&#xff0c;并与他人共享。Hvigor主要提供了两种方式来实现插件&#xff1a;基于hvigorfile脚本开发插件、基于typescript项目开发。下面以基于hvigorfile脚本开发插件进行介绍。 基于hvigorfile脚本…

网络安全-------防止被抓包

1.Ios应用网络安全之https 安全套接字层 (Secure Socket Layer, SSL) 是用来实现互联网安全通信的最普遍的标准。Web 应用程序使用 HTTPS&#xff08;基于 SSL 的 HTTP&#xff09;&#xff0c;HTTPS 使用数字证书来确保在服务器和客户端之间进行安全、加密的通信。在 SSL 连接…

11.20 深度学习-pytorch包和属性的基础语法

import torch import numpy as np def sci_close(): # 关闭pytorch 数据打印出来时科学计数法 torch.set_printoptions(sci_modeFalse) pass return 0 def create_tensor(): # 创建张量 t1torch.tensor(5) # 一阶张量 阶数看你传入的矩阵是多少阶的 这个是标量 不是一阶 一阶…

深入探索JMeter的执行器时间线:从CLArgsParser到JmeterEngine

引言 Apache JMeter是一款广泛使用的开源性能测试工具&#xff0c;它允许用户对各种服务进行负载测试。然而&#xff0c;了解其内部工作机制对于优化测试计划和提高测试效率至关重要。本文将深入探讨JMeter的执行器时间线&#xff0c;包括CLArgsParser、HashTree、StandardJme…

C++结构型设计模式所体现面向接口设计的特征和优点

结构型设计模式&#xff08;Structural Patterns&#xff09;在面向接口设计方面体现了一系列重要的特征&#xff0c;这些特征帮助我们构建灵活、可扩展和易于维护的系统。以下是结构型设计模式在面向接口设计方面的特征及其优点&#xff1a; 1. 接口分离和抽象化 特征 结构…