Android 單例模式的四種實現(xiàn)方式
public class SingletionStarving { private static final SingletionStarving mInstance = new SingletionStarving(); private SingletionStarving() { } public static SingletionStarving getInstance() {return mInstance; }} 構(gòu)造函數(shù)用private修飾,外部無法訪問 聲明靜態(tài)對象時就初始化 static關(guān)鍵字修飾,靜態(tài)變量,存儲在內(nèi)存中,只有一份數(shù)據(jù)。 final關(guān)鍵字,只初始化一次,所以mInstance實例只有一個。二.懶漢式
public class SingletionSlacker { private static SingletionSlacker mInstance; private SingletionSlacker() {} public static synchronized SingletionSlacker getInstance() {if (mInstance == null) { mInstance = new SingletionSlacker();}return mInstance; }} 構(gòu)造函數(shù)用private修飾,外部無法訪問 使用的時候即調(diào)用getInstance的時候才初始化 static關(guān)鍵字修飾,靜態(tài)變量,存儲在內(nèi)存中,只有一份數(shù)據(jù)。 synchronized線程安全,多線程情況下單例的唯一性 缺點(diǎn):沒次調(diào)用getInstance都會同步一次,浪費(fèi)資源 三.雙重檢查加鎖方式
網(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; }} 構(gòu)造函數(shù)用private修飾,外部無法訪問 使用的時候即調(diào)用getInstance的時候才初始化 static關(guān)鍵字修飾,靜態(tài)變量,存儲在內(nèi)存中,只有一份數(shù)據(jù) synchronized線程安全,多線程情況下單例的唯一性 兩次判斷空,避免多次同步(synchronized)
缺點(diǎn)
private static Singletion mInstance;private Singletion() {}public static Singletion getmInstance() {}
由于jvm特性,允許亂序執(zhí)行,上面三句代碼順序不定,那么就可能出現(xiàn)失效的問題。步驟一、倘若A線程執(zhí)行g(shù)etmInstance(),還沒執(zhí)行構(gòu)造方法Singletion()步驟二、此時B線程調(diào)用getmInstance()。因為A已經(jīng)執(zhí)行g(shù)etmInstance(),所以mInstance不為空就直接獲取。步驟三、由于B直接獲取,而真實情況是A線程構(gòu)造方法還未執(zhí)行,所以mInstance就為空了。雖然此情況發(fā)生概率較小,但也是一種情況。為了解決這種情況,java1.6開始加入volatile關(guān)鍵字
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關(guān)鍵字的寫法基本能滿足絕大部分情況。除非你要運(yùn)行在高并發(fā),或者java1.6之前的代碼中。
四.靜態(tài)內(nèi)部類方式public class SingletionInternalClass { private SingletionInternalClass() {} public static SingletionInternalClass getInstance() {return SingletionInternalClassHolder.instance; } private static class SingletionInternalClassHolder {private static final SingletionInternalClass instance = new SingletionInternalClass(); }}
構(gòu)造函數(shù)用private修飾,外部無法訪問
使用的時候即調(diào)用getInstance的時候才初始化
調(diào)用getInstance才回去加載SingletionInternalClassHolder類,確保了線程安全,保證了單例的唯一性
總結(jié)單例模式不管用那種方式實現(xiàn),核心思想都相同1、構(gòu)造函數(shù)私有化,通過一次靜態(tài)方法獲取一個唯一實例2、線程安全
最后推薦使用文中**雙重鎖方式和靜態(tài)內(nèi)部類的方式**來創(chuàng)建單例模式。
以上就是Android 單例模式的四種實現(xiàn)方式的詳細(xì)內(nèi)容,更多關(guān)于Android 單例模式的實現(xiàn)的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. python中scrapy處理項目數(shù)據(jù)的實例分析2. 教你在 IntelliJ IDEA 中使用 VIM插件的詳細(xì)教程3. IntelliJ IDEA導(dǎo)入jar包的方法4. js抽獎轉(zhuǎn)盤實現(xiàn)方法分析5. Python requests庫參數(shù)提交的注意事項總結(jié)6. vue-electron中修改表格內(nèi)容并修改樣式7. iOS實現(xiàn)點(diǎn)贊動畫特效8. 通過Python pyecharts輸出保存圖片代碼實例9. SpringBoot參數(shù)校驗與國際化使用教程10. PHP橋接模式Bridge Pattern的優(yōu)點(diǎn)與實現(xiàn)過程
