Android 單例模式的四種實現(xiàn)方式
一.餓漢式
public class SingletionStarving {
private static final SingletionStarving mInstance = new SingletionStarving();
private SingletionStarving() {
}
public static SingletionStarving getInstance() {
return mInstance;
}
}
- 構造函數(shù)用private修飾,外部無法訪問
- 聲明靜態(tài)對象時就初始化
- static關鍵字修飾,靜態(tài)變量,存儲在內存中,只有一份數(shù)據(jù)。
- final關鍵字,只初始化一次,所以mInstance實例只有一個。
二.懶漢式
public class SingletionSlacker {
private static SingletionSlacker mInstance;
private SingletionSlacker() {}
public static synchronized SingletionSlacker getInstance() {
if (mInstance == null) {
mInstance = new SingletionSlacker();
}
return mInstance;
}
}
- 構造函數(shù)用private修飾,外部無法訪問
- 使用的時候即調用getInstance的時候才初始化
- static關鍵字修飾,靜態(tài)變量,存儲在內存中,只有一份數(shù)據(jù)。
- synchronized線程安全,多線程情況下單例的唯一性
- 缺點:沒次調用getInstance都會同步一次,浪費資源
三.雙重檢查加鎖方式
網(wǎng)上建議和使用最多的方法
public class Singletion {
private static Singletion mInstance;
private Singletion() {}
public static Singletion getmInstance() {
if (mInstance == null) {
synchronized (Singletion.class) { if (mInstance == null) {
mInstance = new Singletion (); }
}
}
return mInstance;
}
}
- 構造函數(shù)用private修飾,外部無法訪問
- 使用的時候即調用getInstance的時候才初始化
- static關鍵字修飾,靜態(tài)變量,存儲在內存中,只有一份數(shù)據(jù)
- synchronized線程安全,多線程情況下單例的唯一性
- 兩次判斷空,避免多次同步(synchronized)
缺點
private static Singletion mInstance;
private Singletion() {}
public static Singletion getmInstance() {}
由于jvm特性,允許亂序執(zhí)行,上面三句代碼順序不定,那么就可能出現(xiàn)失效的問題。
步驟一、倘若A線程執(zhí)行getmInstance(),還沒執(zhí)行構造方法Singletion()
步驟二、此時B線程調用getmInstance()。因為A已經(jīng)執(zhí)行getmInstance(),所以mInstance不為空就直接獲取。
步驟三、由于B直接獲取,而真實情況是A線程構造方法還未執(zhí)行,所以mInstance就為空了。
雖然此情況發(fā)生概率較小,但也是一種情況。為了解決這種情況,java1.6開始加入volatile關鍵字
private volatile static Singletion mInstance;
這樣就避免了方式失效的情況。雖然會volatile消耗一些性能,所以最佳寫法
public class Singletion {
private volatile static Singletion mInstance;
private Singletion () {}
public static Singletion getmInstance() { if (mInstance == null) {
synchronized (Singletion.class) { if (mInstance == null) {
mInstance = new Singletion(); }
}
}
return mInstance;
}
}
雖然volatile讓方式完美,但是沒有volatile關鍵字的寫法基本能滿足絕大部分情況。除非你要運行在高并發(fā),或者java1.6之前的代碼中。
四.靜態(tài)內部類方式
public class SingletionInternalClass {
private SingletionInternalClass() {}
public static SingletionInternalClass getInstance() {
return SingletionInternalClassHolder.instance;
}
private static class SingletionInternalClassHolder {
private static final SingletionInternalClass instance = new SingletionInternalClass();
}
}
構造函數(shù)用private修飾,外部無法訪問
使用的時候即調用getInstance的時候才初始化
調用getInstance才回去加載SingletionInternalClassHolder類,確保了線程安全,保證了單例的唯一性
總結
單例模式不管用那種方式實現(xiàn),核心思想都相同
1、構造函數(shù)私有化,通過一次靜態(tài)方法獲取一個唯一實例
2、線程安全
最后推薦使用文中**雙重鎖方式和靜態(tài)內部類的方式**來創(chuàng)建單例模式。
以上就是Android 單例模式的四種實現(xiàn)方式的詳細內容,更多關于Android 單例模式的實現(xiàn)的資料請關注腳本之家其它相關文章!
相關文章
在Ubuntu下搭建Android開發(fā)環(huán)境
對一個程序猿來說,裝好系統(tǒng)之后的第一件事,一定是搭建開發(fā)環(huán)境,已經(jīng)安裝各種開發(fā)工具,以便之后能方便順利地進行程序的開發(fā)。簡單的介紹下在Ubuntu環(huán)境下搭建Android開發(fā)環(huán)境,雖然基本上和在Windows下沒有太大差別,但有些細節(jié)上還是很值得注意的。2014-07-07
Android開發(fā)實現(xiàn)ImageView加載攝像頭拍攝的大圖功能
這篇文章主要介紹了Android開發(fā)實現(xiàn)ImageView加載攝像頭拍攝的大圖功能,涉及Android基于ImageView的攝像頭拍攝圖片加載、保存及權限控制等相關操作技巧,需要的朋友可以參考下2017-11-11
Android ListView下拉刷新上拉自動加載更多DEMO示例
這篇文章主要介紹了Android ListView下拉刷新上拉自動加載更多DEMO示例的相關資料,非常不錯,具有參考借鑒價值,需要的朋友可以參考下2016-07-07
Android 通過當前經(jīng)緯度獲得城市的實例代碼
Android 通過當前經(jīng)緯度獲得城市的實例代碼,需要的朋友可以參考一下2013-06-06
Android 自定義View結合自定義TabLayout實現(xiàn)頂部標簽滑動效果
小編最近在做app的項目,需要用到tablayout實現(xiàn)頂部的滑動效果,文中代碼用到了自定義item,代碼也很簡單,感興趣的朋友跟隨腳本之家小編一起看看吧2018-07-07
Android開發(fā)中TextView文本過長滾動顯示實現(xiàn)方法分析
這篇文章主要介紹了Android開發(fā)中TextView文本過長滾動顯示實現(xiàn)方法,結合實例形式分析了Android項目開發(fā)中TextView顯示超長文本的具體操作技巧與注意事項,需要的朋友可以參考下2018-02-02

