众所周知,java中是JVM负责内存的分配和回收,这是它的优势(使用方便,程序不用再像使用c那样操心内存),但同时也是它的缺点(不够灵活)。为了解决内存操做不灵活这个问题,能够采用软引用等方法。java
在JDK1.2之前的版本中,当一个对象不被任何变量引用,那么程序就没法再使用这个对象。也就是说,只有对象处于可触及状态,程序才能使用它。这 就像在平常生活中,从商店购买了某样物品后,若是有用,就一直保留它,不然就把它扔到垃圾箱,由清洁工人收走。通常说来,若是物品已经被扔到垃圾箱,想再 把它捡回来使用就不可能了。缓存
但有时候状况并不这么简单,你可能会遇到相似鸡肋同样的物品,食之无味,弃之惋惜。这种物品如今已经无用了,保留它会占空间,可是马上扔掉它也不划算,因 为也许未来还会派用场。对于这样的无关紧要的物品,一种折衷的处理办法是:若是家里空间足够,就先把它保留在家里,若是家里空间不够,即便把家里全部的垃 圾清除,仍是没法容纳那些必不可少的生活用品,那么再扔掉这些无关紧要的物品。
从JDK1.2版本开始,把对象的引用分为四种级别,从而使程序能更加灵活的控制对象的生命周期。这四种级别由高到低依次为:强引用、软引用、弱引用和虚引用。安全
1.强引用spa
之前咱们使用的大部分引用实际上都是强引用,这是使用最广泛的引用。若是一个对象具备强引用,那就相似于必不可少的生活用品,垃圾回收器毫不会回收它。当内存空 间不足,Java虚拟机宁愿抛出OutOfMemoryError错误,使程序异常终止,也不会靠随意回收具备强引用的对象来解决内存不足问题。线程
2.软引用(SoftReference)设计
若是一个对象只具备软引用,那就相似于无关紧要的生活用品。若是内存空间足够,垃圾回收器就不会回收它,若是内存空间不足了,就会回收这些对象的内存。只要垃圾回收器没有回收它,该对象就能够被程序使用。软引用可用来实现内存敏感的高速缓存。
软引用能够和一个引用队列(ReferenceQueue)联合使用,若是软引用所引用的对象被垃圾回收,JAVA虚拟机就会把这个软引用加入到与之关联的引用队列中。code
3.弱引用(WeakReference)
若是一个对象只具备弱引用,那就相似于可有可物的生活用品。弱引用与软引用的区别在于:只具备弱引用的对象拥有更短暂的生命周期。在垃圾回收器线程扫描它 所管辖的内存区域的过程当中,一旦发现了只具备弱引用的对象,无论当前内存空间足够与否,都会回收它的内存。不过,因为垃圾回收器是一个优先级很低的线程, 所以不必定会很快发现那些只具备弱引用的对象。
弱引用能够和一个引用队列(ReferenceQueue)联合使用,若是弱引用所引用的对象被垃圾回收,Java虚拟机就会把这个弱引用加入到与之关联的引用队列中。对象
4.虚引用(PhantomReference)
"虚引用"顾名思义,就是形同虚设,与其余几种引用都不一样,虚引用并不会决定对象的生命周期。若是一个对象仅持有虚引用,那么它就和没有任何引用同样,在任什么时候候均可能被垃圾回收。
虚引用主要用来跟踪对象被垃圾回收的活动。虚引用与软引用和弱引用的一个区别在于:虚引用必须和引用队列(ReferenceQueue)联合使用。当垃 圾回收器准备回收一个对象时,若是发现它还有虚引用,就会在回收对象的内存以前,把这个虚引用加入到与之关联的引用队列中。程序能够经过判断引用队列中是 否已经加入了虚引用,来了解blog
被引用的对象是否将要被垃圾回收。程序若是发现某个虚引用已经被加入到引用队列,那么就能够在所引用的对象的内存被回收以前采起必要的行动。生命周期
特别注意,在世纪程序设计中通常不多使用弱引用与虚引用,使用软用的状况较多,这是由于软引用能够加速JVM对垃圾内存的回收速度,能够维护系统的运行安全,防止内存溢出(OutOfMemory)等问题的产生。
如下是软引用的代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
import
java.lang.ref.SoftReference;
public
class
Test {
public
static
void
main(String[] args){
System.out.println(
"开始"
);
A a =
new
A();
SoftReference<A> sr =
new
SoftReference<A>(a);
a =
null
;
if
(sr!=
null
){
a = sr.get();
}
else
{
a =
new
A();
sr =
new
SoftReference<A>(a);
}
System.out.println(
"结束"
);
}
}
class
A{
int
[] a ;
public
A(){
a =
new
int
[
100000000
];
}
}
|