Java的四種引用方式
java內存管理分為內存分配和內存回收,都不需要程序員負責,垃圾回收的機制主要是看對象是否有引用指向該對象。
java對象的引用包括
強引用,軟引用,弱引用,虛引用
Java中提供這四種引用類型主要有兩個目的:
第一是可以讓程序員通過代碼的方式決定某些對象的生命周期;
第二是有利於JVM進行垃圾回收。
下面來闡述一下這四種類型引用的概念:
1.強引用
是指創建一個對象並把這個對象賦給一個引用變量。
比如:
Object object =
new
Object();
String str =
"hello"
;
強引用有引用變量指向時永遠不會被垃圾回收,JVM寧願拋出OutOfMemory錯誤也不會回收這種對象。
- <pre name="code" class="java">public class Main {
- public static void main(String[] args) {
- new Main().fun1();
- }
- public void fun1() {
- Object object = new Object();
- Object[] objArr = new Object[1000];
- }
當運行至Object[] objArr = new Object[1000];這句時,如果內存不足,JVM會拋出OOM錯誤也不會回收object指向的對象。不過要注意的是,當fun1運行完之后,object和objArr都已經不存在了,所以它們指向的對象都會被JVM回收。
如果想中斷強引用和某個對象之間的關聯,可以顯示地將引用賦值為null,這樣一來的話,JVM在合適的時間就會回收該對象。
比如Vector類的clear方法中就是通過將引用賦值為null來實現清理工作的:
- <pre name="code" class="java">/**
- * Removes the element at the specified position in this Vector.
- * Shifts any subsequent elements to the left (subtracts one from their
- * indices). Returns the element that was removed from the Vector.
- *
- * @throws ArrayIndexOutOfBoundsException if the index is out of range
- * ({@code index < 0 || index >= size()})
- * @param index the index of the element to be removed
- * @return element that was removed
- * @since 1.2
- */
- public synchronized E remove(int index) {
- modCount++;
- if (index >= elementCount)
- throw new ArrayIndexOutOfBoundsException(index);
- Object oldValue = elementData[index];
- int numMoved = elementCount - index - 1;
- if (numMoved > 0)
- System.arraycopy(elementData, index+1, elementData, index,
- numMoved);
- elementData[--elementCount] = null; // Let gc do its work
- return (E)oldValue;
- }
2.軟引用(SoftReference)
如果一個對象具有軟引用,內存空間足夠,垃圾回收器就不會回收它;
如果內存空間不足了,就會回收這些對象的內存。只要垃圾回收器沒有回收它,該對象就可以被程序使用。
軟引用可用來實現內存敏感的高速緩存,比如網頁緩存、圖片緩存等。使用軟引用能防止內存泄露,增強程序的健壯性。
SoftReference的特點是它的一個實例保存對一個Java對象的軟引用, 該軟引用的存在不妨礙垃圾收集線程對該Java對象的回收。
也就是說,一旦SoftReference保存了對一個Java對象的軟引用后,在垃圾線程對 這個Java對象回收前,SoftReference類所提供的get()方法返回Java對象的強引用。
另外,一旦垃圾線程回收該Java對象之 后,get()方法將返回null。
舉個栗子:
- <pre name="code" class="java">MyObject aRef = new MyObject();
- SoftReference aSoftRef=new SoftReference(aRef);
此時,對於這個MyObject對象,有兩個引用路徑,一個是來自SoftReference對象的軟引用,一個來自變量aReference的強引用,所以這個MyObject對象是強可及對象。
- <pre name="code" class="java">aRef = null;
Java虛擬機的垃圾收集線程對軟可及對象和其他一般Java對象進行了區別對待:軟可及對象的清理是由垃圾收集線程根據其特定算法按照內存需求決定的。
也就是說,垃圾收集線程會在虛擬機拋出OutOfMemoryError之前回收軟可及對象,而且虛擬機會盡可能優先回收長時間閑置不用的軟可及對象,對那些剛剛構建的或剛剛使用過的“新”軟可反對象會被虛擬機盡可能保留。在回收這些對象之前,我們可以通過:
- <pre name="code" class="java">MyObject anotherRef=(MyObject)aSoftRef.get();
重新獲得對該實例的強引用。而回收之后,調用get()方法就只能得到null了。
使用ReferenceQueue清除失去了軟引用對象的SoftReference:
- <pre name="code" class="java">ReferenceQueue queue = new ReferenceQueue();
- SoftReference ref=new SoftReference(aMyObject, queue);
那么當這個SoftReference所軟引用的aMyOhject被垃圾收集器回收的同時,ref所強引用的SoftReference對象被列入ReferenceQueue。也就是說,ReferenceQueue中保存的對象是Reference對象,而且是已經失去了它所軟引用的對象的Reference對象。另外從ReferenceQueue這個名字也可以看出,它是一個隊列,當我們調用它的poll()方法的時候,如果這個隊列中不是空隊列,那么將返回隊列前面的那個Reference對象。
- <pre name="code" class="java">SoftReference ref = null;
- while ((ref = (EmployeeRef) q.poll()) != null) {
- // 清除ref
- }
3.弱引用(WeakReference)
弱引用也是用來描述非必需對象的,當JVM進行垃圾回收時,無論內存是否充足,都會回收被弱引用關聯的對象。在java中,用java.lang.ref.WeakReference類來表示。下面是使用示例:
- <pre name="code" class="java">public class test {
- public static void main(String[] args) {
- WeakReference<People>reference=new WeakReference<People>(new People("zhouqian",20));
- System.out.println(reference.get());
- System.gc();//通知GVM回收資源
- System.out.println(reference.get());
- }
- }
- class People{
- public String name;
- public int age;
- public People(String name,int age) {
- this.name=name;
- this.age=age;
- }
- @Override
- public String toString() {
- return "[name:"+name+",age:"+age+"]";
- }
- }
- 輸出結果:
[name:zhouqian,age:20]
null
第二個輸出結果是null,這說明只要JVM進行垃圾回收,被弱引用關聯的對象必定會被回收掉。不過要注意的是,這里所說的被弱引用關聯的對象是指只有弱引用與之關聯,如果存在強引用同時與之關聯,則進行垃圾回收時也不會回收該對象(軟引用也是如此)。
比如:將代碼做一點小更改:
- <pre name="code" class="java">package yinyong;
- import java.lang.ref.WeakReference;
- public class test {
- public static void main(String[] args) {
- People people=new People("zhouqian",20);
- WeakReference<People>reference=new WeakReference<People>(people);//<span style="color:#FF0000;">關聯強引用</span>
- System.out.println(reference.get());
- System.gc();
- System.out.println(reference.get());
- }
- }
- class People{
- public String name;
- public int age;
- public People(String name,int age) {
- this.name=name;
- this.age=age;
- }
- @Override
- public String toString() {
- return "[name:"+name+",age:"+age+"]";
- }
- }//結果發生了很大的變化
- [name:zhouqian,age:20]
- [name:zhouqian,age:20]
弱引用可以和一個引用隊列(ReferenceQueue)聯合使用,如果弱引用所引用的對象被JVM回收,這個軟引用就會被加入到與之關聯的引用隊列中。
4.虛引用(PhantomReference)
虛引用和前面的軟引用、弱引用不同,它並不影響對象的生命周期。在java中用java.lang.ref.PhantomReference類表示。如果一個對象與虛引用關聯,則跟沒有引用與之關聯一樣,在任何時候都可能被垃圾回收器回收。
要注意的是,虛引用必須和引用隊列關聯使用,當垃圾回收器准備回收一個對象時,如果發現它還有虛引用,就會把這個虛引用加入到與之 關聯的引用隊列中。程序可以通過判斷引用隊列中是否已經加入了虛引用,來了解被引用的對象是否將要被垃圾回收。如果程序發現某個虛引用已經被加入到引用隊列,那么就可以在所引用的對象的內存被回收之前采取必要的行動。
- <pre name="code" class="java">import java.lang.ref.PhantomReference;
- import java.lang.ref.ReferenceQueue;
- public class Main {
- public static void main(String[] args) {
- ReferenceQueue<String> queue = new ReferenceQueue<String>();
- PhantomReference<String> pr = new PhantomReference<String>(new String("hello"), queue);
- System.out.println(pr.get());
- }
- }
軟引用和弱引用
對於強引用,我們平時在編寫代碼時經常會用到。而對於其他三種類型的引用,使用得最多的就是軟引用和弱引用,這2種既有相似之處又有區別。它們都是用來描述非必需對象的,但是被軟引用關聯的對象只有在內存不足時才會被回收,而被弱引用關聯的對象在JVM進行垃圾回收時總會被回收。
在SoftReference類中,有三個方法,兩個構造方法和一個get方法(WekReference類似):
兩個構造方法:
- <pre name="code" class="java">public SoftReference(T referent) {
- super(referent);
- this.timestamp = clock;
- }
- public SoftReference(T referent, ReferenceQueue<? super T> q) {
- super(referent, q);
- this.timestamp = clock;
- }
get方法用來獲取與軟引用關聯的對象的引用,如果該對象被回收了,則返回null。
在使用軟引用和弱引用的時候,我們可以顯示地通過System.gc()來通知JVM進行垃圾回收,但是要注意的是,雖然發出了通知,JVM不一定會立刻執行,也就是說這句是無法確保此時JVM一定會進行垃圾回收的。
對象可及性的判斷
在很多時候,一個對象並不是從根集直接引用的,而是一個對象被其他對象引用,甚至同時被幾個對象所引用,從而構成一個以根集為頂的樹形結構。如圖2所示

如何利用軟引用和弱引用解決OOM問題
前面講了關於軟引用和弱引用相關的基礎知識,那么到底如何利用它們來優化程序性能,從而避免OOM的問題呢?
下面舉個例子,假如有一個應用需要讀取大量的本地圖片,如果每次讀取圖片都從硬盤讀取,則會嚴重影響性能,但是如果全部加載到內存當中,又有可能造成內存溢出,此時使用軟引用可以解決這個問題。
設計思路是:用一個HashMap來保存圖片的路徑 和 相應圖片對象關聯的軟引用之間的映射關系,在內存不足時,JVM會自動回收這些緩存圖片對象所占用的空間,從而有效地避免了OOM的問題。在Android開發中對於大量圖片下載會經常用到。
MyObject aRef =
new MyObject();
SoftReference aSoftRef=
new SoftReference(aRef);
|
aRef = null; |
MyObject anotherRef=(MyObject)aSoftRef.get(); |
ReferenceQueue queue =
new ReferenceQueue();
SoftReference ref=
new SoftReference(aMyObject, queue);
|
SoftReference ref =
null;
while ((ref = (EmployeeRef) q.poll()) !=
null) {
// 清除ref
}
|
publicclass Employee {
private String id;// 雇員的標識號碼
private String name;// 雇員姓名
private String department;// 該雇員所在部門
private String Phone;// 該雇員聯系電話
privateintsalary;// 該雇員薪資
private String origin;// 該雇員信息的來源
// 構造方法
public Employee(String id) {
this.id = id;
getDataFromlnfoCenter();
}
// 到數據庫中取得雇員信息
privatevoid getDataFromlnfoCenter() {
// 和數據庫建立連接井查詢該雇員的信息,將查詢結果賦值
// 給name,department,plone,salary等變量
// 同時將origin賦值為"From DataBase"
}
……
|
publicclass SocketManager {
private Map<Socket, User> m =
new HashMap<Socket, User>();
publicvoid setUser(Socket s, User u) {
m.put(s, u);
}
public User getUser(Socket s) {
returnm.get(s);
}
publicvoid removeUser(Socket s) {
m.remove(s);
}
}
|
import java.util.WeakHashMap;
class Element {
private String ident;
public Element(String id) {
ident = id;
}
public String toString() {
returnident;
}
publicint hashCode() {
returnident.hashCode();
}
publicboolean equals(Object obj) {
return obj
instanceof Element && ident.equals(((Element) obj).ident);
}
protectedvoid finalize(){
System.
out.println("Finalizing "+getClass().getSimpleName()+" "+ident);
}
}
class Key
extends Element{
public Key(String id){
super(id);
}
}
class Value
extends Element{
public Value (String id){
super(id);
}
}
publicclass CanonicalMapping {
publicstaticvoid main(String[] args){
int size=1000;
Key[] keys=
new Key[size];
WeakHashMap<Key,Value> map=
new WeakHashMap<Key,Value>();
for(
int i=0;i<size;i++){
Key k=
new Key(Integer.
toString(i));
Value v=
new Value(Integer.
toString(i));
if(i%3==0)
keys[i]=k;
map.put(k, v);
}
System.
gc();
}
}
|
publicclass SocketManager {
private Map<Socket,User> m =
new WeakHashMap<Socket,User>();
publicvoid setUser(Socket s, User u) {
m.put(s, u);
}
public User getUser(Socket s) {
returnm.get(s);
}
}
|

