概述
一.简介
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。也就是说 无法通过虚引用来获取对象的真实引用。所以虚引用一般不会使用。
最后
以上就是健忘大碗为你收集整理的对象引用 强引用+软引用+弱引用+虚引用详解的全部内容,希望文章能够帮你解决对象引用 强引用+软引用+弱引用+虚引用详解所遇到的程序开发问题。
如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。
发表评论 取消回复