弱引用该怎么用?

简介: 弱引用该怎么用?

在工作实践中发现,大家写的代码很容易发生内存泄漏,我觉得这个主要问题是大家对弱引用和 gc root 的理解不够深导致,所以,打算(水)一篇我的理解。


在 维基百科中对弱引用的解释是:


计算机程序设计中,弱引用强引用相对,是指不能确保其引用的对象不会被垃圾回收器回收的引用。一个对象若只被弱引用所引用,则被认为是不可访问(或弱可访问)的,并因此可能在任何时刻被回收


这个意思也很简单,简单翻译一下:


默认我们 new 创建的对象都是强引用,我们可以创建个弱引用来关联这个引用对象,但如果这个对象没有被强引用,只剩下弱引用与他有关联,则认为在任何时刻都可能被回收。 那什么情况下强引用不与对象产生关联呢?那就是 gc 回收时,也即意味着强引用被回收了,则弱引用关联的强引用就会变为不可访问的引用(这句话很重要)。


谁可以作为 gc root?


引用自《深入理解 Java 虚拟机》:

1、在虚拟机栈(栈帧中的本地变量表),譬如各个线程被调用的方法堆栈中使用到的参数、局部变量、临时变量等。

2、在方法区中类静态属性引用的对象,譬如 java 类的引用类型静态变量。

3、在方法区中常量引用的对象,譬如字符串常量池里的引用。

4、在本地方法栈中 JNI 引用的对象。

5、Java 虚拟机内部的引用,如基本数据类型对应的 class 对象,一些常驻的异常对象等,还有类加载器。

6、所有被同步锁持有的对象。

7、反映 Java 虚拟机内部情况的 JMXBean、JVMTI 中注册的回调、本地代码缓存等。


作为 gc root 的条件有很多,下面拿 2 和 5 这两种 gc root 来讲解


例子:


// Test.java
public class Test {
  private static final Test s = new Test(); 
  public static Test getInstance(){
      return s;
  }
  Context context;
  public void setContext(Context context){
       this.context = context;
  }
}
// TestActivity.java
public class TestActivity extends AppCompatActivity{
    public void onCreate(Bundle onSaveInstance){
        Test.getInstance().setContext(this)
    }
}
复制代码


在启动 TestActivity 的时候,onCreate 方法将当前的 Context 引用设置给 Test,我们来画个引用链图:


image.png

在 TestActivity 进行页面销毁时,会执行 ActivityThread 的  performDestroyActivity 方法,该方法会把 TestActivity 从 mActivities 集合中移除,也就是断开引用,使其能被 gc 回收掉,执行之后的图:

image.png

可惜的是,TestActivity 虽然从 ActivityThread 的 gc root 引用链中被移除,但又被 Test 的 gc root 给强引用,导致 TestActivity 的整个引用链无法被释放而出现内存泄漏。


这时候就要用弱引用去解,Test 不去强引用 TestActivity,而是通过弱引用去关联强引用,如果强引用被回收的话,则弱引用也变为不可访问,也即我们通常遇到的 weakRef.get() 是空的情况,我们需要改一下 Test 的 setContext 方法:


Context context;
 public void setContext(Context context){
   WeakReference<Context> contextRef = new WeakReference<>(context);
   context = contextRef.get();
 }
复制代码


这时,我们的引用链图如下:


image.png

TestActivity 没有被任何依赖强引用,在 gc 到来时,TestActivity 则会被回收,Test 类中持有的弱引用 context 值会变成空,最终为:


image.png

总结:



如上只是用单例这种情况去做了一个分析,当然还有匿名内部类持有外部引用这种,如 Handler,道理都是一样,万变不离其宗,对于 Handler 的引用链分析可以看虾哥的文章《一次性讲清楚 Handler 可能导致的内存泄漏和解决办法


对于弱引用怎么用,就是要看在 2 个及以上的 gc root 中是否都对同一个对象持有引用,如果有,则可以用弱引用去隔离开

目录
相关文章
|
5月前
|
人工智能 运维 架构师
技能革命3.0时代:云计算就业岗位有哪些?
本文探讨云计算就业市场的深层逻辑,从岗位体系、AI赋能及技术局限性应对策略三方面解析。云计算岗位涵盖基础设施、平台服务、数据智能与应用创新四层,需复合型能力;AI不仅提供技术工具,还推动教育范式变革,助力跨界融合;面对技术局限,分步验证与经验洞察双管齐下。未来就业将向技能多元化、自主性增强和社会价值再定义方向进化,强调个体能力生态的持续成长。
|
canal Kubernetes Cloud Native
云原生|kubernetes|kubeadm部署高可用集群(一)使用外部etcd集群(二)
云原生|kubernetes|kubeadm部署高可用集群(一)使用外部etcd集群
654 0
神奇的「小黄鸭调试法」
什么意思?就是说,当你被一段代码整不会了,怎么看也看不出问题出在哪里的时候。你可以对着一只小黄鸭,一行一行解释你的代码。不仅解释代码的功能,你最好还要告诉鸭子你到底想实现什么目标,以及你打算通过什么样的方法实现,为什么要这么实现等等。当你说着说着说着,哎,突然你就明白问题在哪了。就是这么的神奇。
|
10月前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
401 4
|
人工智能 自然语言处理 搜索推荐
AI战略丨生成式人工智能应用场景与落地路径
GenAI 的应用落地是一项系统性工程,离不开战略、业务、技术、组织等多维度的统筹和推动。
|
监控 Java 编译器
jstat使用实用教程
jstat使用实用教程
190 0
|
存储 算法 安全
深入理解C++迭代器:让你的C++代码更加灵活
深入理解C++迭代器:让你的C++代码更加灵活
311 0
|
编译器 C++ 开发者
解读C++ constexpr关键字的特性
解读C++ constexpr关键字的特性
233 0
解读C++ constexpr关键字的特性
|
存储 缓存 Java
WeakMap 和 WeakSet:解决内存泄漏&避免循环引用(下)
WeakMap 和 WeakSet:解决内存泄漏&避免循环引用(下)
WeakMap 和 WeakSet:解决内存泄漏&避免循环引用(下)
|
安全 算法 编译器
C++中=delete的巧妙用法
C++中=delete的巧妙用法
509 0