对象引用 强引用+软引用+弱引用+虚引用详解

一.简介

 

Java中,为了使程序能更灵活地控制对象的生命周期,从 JDK 1.2 版本开始,JDK对象的 引用级别 由 高到低 分为 强引用软引用弱引用虚引用 四种级别

 

Java的内存分配和内存回收,都不需要程序员负责,而是由JVM去负责。一个对象是否可以被回收,主要看是否有引用指向此对象

 

Java设计这四种引用的主要目的有两个。

   <1> 方便程序员通过代码的方式来决定某个对象的生命周期。

   <2> 有利于垃圾回收。

 

下面具体讲解

 

 

 

 

 

 

 

二.强引用

 

1.简介

StrongReference

强引用是使用最普遍的引用。如果一个对象具有强引用,那垃圾回收器绝不会回收它。当内存空间不足,Java虚拟机宁愿抛出OutOfMemoryError错误,使程序异常终止,也不会靠随意回收具有强引用的对象来解决内存不足的问题。在项目中估计99.99%的代码都是强引用。 

 

 

2.举例

People people = new People();
        
String s = new String("abc");

...

Object o = new Object();

 

 

 

 

 

 

三.软引用

 

1.简介

SoftReference

如果一个对象只具有软引用,当内存空间足够,垃圾回收器就不会回收它。当内存空间不足,就会回收这些对象的内存。只要垃圾回收器没有回收它,该对象就可以被程序使用

软引用可用来实现内存敏感的高速缓存。软引用可以和一个引用队列(ReferenceQueue)联合使用,如果软引用所引用的对象被垃圾回收器回收,Java虚拟机就会把这个软引用加入到与之关联的引用队列中。

 

 

2.举例

 

<1> 简单举例

//声明
private SoftReference<HandlerActivity> activitySoftReference;

...

//适当时机 创建SoftReference对象 并将要存储的对象 存储到SoftReference中
activitySoftReference = new SoftReference<>(handlerActivity);

...

//get()方法获取SoftReference存储的对象 
HandlerActivity activity = activitySoftReference.get();

//因为SoftReference对象有可能被回收,所以使用的时候需要判断存储的对象是否为空
if (null != activity) {
    activity.setTextView("123");
}

...

 

 

<2> Bitmap

 Android 图片库之自定义ImageLoader类

 

 

 

 

四.弱引用

 

1.简介

WeakReference

弱引用软引用的区别在于:只具有弱引用的对象拥有更短暂的生命周期在垃圾回收器线程扫描它所管辖的内存区域的过程中,一旦发现了只具有弱引用的对象,不管当前内存空间足够与否,都会回收它的内存。不过,由于垃圾回收器是一个优先级很低的线程,因此不一定会很快发现那些只具有弱引用的对象。


弱引用可以和一个引用队列(ReferenceQueue)联合使用,如果弱引用所引用的对象被垃圾回收,Java虚拟机就会把这个弱引用加入到与之关联的引用队列中。

 

 

2.举例

package com.example.myapplication.handler;
 
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.util.Log;
import android.widget.TextView;
 
import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
 
import com.example.myapplication.R;
 
import java.lang.ref.WeakReference;
 
public class HandlerActivity extends AppCompatActivity {
 
    private TextView textView;
 
    private Handler mHandler = new MyHandler(this);
 
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_handler);
        findView();
    }
 
    /**
     * 初始化各种View
     */
 
    private void findView() {
        textView = findViewById(R.id.activity_handler_textview);
        newThreadMethod();
    }
 
    /**
     * Handler静态内部类
     */
 
    private static class MyHandler extends Handler {
 
        private WeakReference<HandlerActivity> activityWeakReference;
 
        public MyHandler(HandlerActivity handlerActivity) {
            if (null == handlerActivity) {
                return;
            }
            activityWeakReference = new WeakReference<>(handlerActivity);
        }
 
        @Override
        public void handleMessage(@NonNull Message msg) {
            super.handleMessage(msg);
            if (null != activityWeakReference) {
                HandlerActivity handlerActivity = activityWeakReference.get();
                if (null != handlerActivity) {
                    switch (msg.what) {
                        case 3:
                            int num1 = msg.arg1;
                            int num2 = msg.arg2;
                            String string = (String) msg.obj;
                            String result = "内部静态类方式声明Handler获取num1----:" + num1 + "  num2----:" + num2 + "  Msg----:" + string;
                            handlerActivity.setTextView(result);
                            break;
                    }
                }
            }
        }
    }
 
    /**
     * 模拟子线程
     */
 
    private void newThreadMethod() {
        if (null == mHandler) {
            return;
        }
 
        new Thread(new Runnable() {
            @Override
            public void run() {
                Message msg = Message.obtain();
                msg.what = 3;
                msg.arg1 = 33;
                msg.arg2 = 333;
                msg.obj = "子线程发送的消息";
                mHandler.sendMessage(msg);
            }
        }).start();
    }
 
    /**
     * 模拟Handler中调用Activity方法
     */
 
    private void setTextView(String result) {
        textView.setText(result);
        Log.d("TAG", result);
    }
 
    /**
     * onDestroy方法
     */
 
    @Override
    protected void onDestroy() {
        super.onDestroy();
        if (mHandler != null) {
            mHandler.removeCallbacksAndMessages(null);
            mHandler = null;
        }
    }
}

 

说明:由三和四分类可知,软引用和弱引用,用法差不多。都有XXX.get()方法获取XXX对象,然后判断XXX对象是否为空。不为空时使用XXX对象。

 

 

 

 

 

五.虚引用

 

1.简介

PhantomReference:虚引用”顾名思义,就是形同虚设,与其他几种引用都不同,虚引用并不会决定对象的生命周期。如果一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收器回收。


虚引用主要用来跟踪对象被垃圾回收器回收的活动。虚引用与软引用和弱引用的一个区别在于:虚引用必须和引用队列 (ReferenceQueue)联合使用。当垃圾回收器准备回收一个对象时,如果发现它还有虚引用,就会在回收对象的内存之前,把这个虚引用加入到与之 关联的引用队列中。

 

2.举例

因为虚引用PhantomReference的get()方法会永远返回null。也就是说 无法通过虚引用来获取对象的真实引用。所以虚引用一般不会使用。

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值