自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Android設(shè)計(jì)模式之單例模式

移動(dòng)開發(fā) Android
一般單例模式包含了5種寫法,分別是餓漢、懶漢、雙重校驗(yàn)鎖、靜態(tài)內(nèi)部類和枚舉。相信看完之后你對(duì)單例模式有了充分的理解了,根據(jù)不同的場景選擇最你最喜歡的一種單例模式吧!

經(jīng)常有人問我說Android學(xué)習(xí)如何進(jìn)階?不管你怎么走,設(shè)計(jì)模式可謂是進(jìn)階必備,對(duì)設(shè)計(jì)模式的理解與運(yùn)用對(duì)你之后的代碼書寫與架構(gòu)設(shè)計(jì)有很多的幫助作用,那么從今天開始我就抽時(shí)間來給大家分享下設(shè)計(jì)模式系列。

[[164566]]

什么是設(shè)計(jì)模式?其實(shí)簡單的理解就是前人留下來的一些經(jīng)驗(yàn)總結(jié)而已,然后把這些經(jīng)驗(yàn)起了個(gè)名字叫Design Pattern,翻譯過來就是設(shè)計(jì)模式的意思,通過使用設(shè)計(jì)模式可以讓我們的代碼復(fù)用性更高,可維護(hù)性更高,讓你的代碼寫的更優(yōu)雅。設(shè)計(jì)模式理論上有23種,但是我只會(huì)針對(duì)Android平臺(tái)上常用的一些設(shè)計(jì)模式做分享,今天就先來分享下最常用的單例模式。
餓漢式

  1. public class Singleton{  
  2.  
  3.     private static Singleton instance = new Singleton();  
  4.  
  5.     private Singleton(){}  
  6.  
  7.     public static Singleton newInstance(){  
  8.         return instance;  
  9.     }  

餓漢式是最簡單的實(shí)現(xiàn)方式,這種實(shí)現(xiàn)方式適合那些在初始化時(shí)就要用到單例的情況,這種方式簡單粗暴,如果單例對(duì)象初始化非??欤艺加脙?nèi)存非常小的時(shí)候這種方式是比較合適的,可以直接在應(yīng)用啟動(dòng)時(shí)加載并初始化。 但是,如果單例初始化的操作耗時(shí)比較長而應(yīng)用對(duì)于啟動(dòng)速度又有要求,或者單例的占用內(nèi)存比較大,再或者單例只是在某個(gè)特定場景的情況下才會(huì)被使用,而一般情況下是不會(huì)使用時(shí),使用 餓漢式 的單例模式就是不合適的,這時(shí)候就需要用到 懶漢式 的方式去按需延遲加載單例。

懶漢式

  1. public class Singleton{  
  2.     private static Singleton instance = null;  
  3.  
  4.     private Singleton(){}  
  5.  
  6.     public static newInstance(){  
  7.         if(null == instance){  
  8.             instance = new Singleton();  
  9.         }  
  10.         return instance;  
  11.     }  

懶漢式與 餓漢式 的***區(qū)別就是將單例的初始化操作,延遲到需要的時(shí)候才進(jìn)行,這樣做在某些場合中有很大用處。比如某個(gè)單例用的次數(shù)不是很多,但是這個(gè)單例提供的功能又非常復(fù)雜,而且加載和初始化要消耗大量的資源,這個(gè)時(shí)候使用 懶漢式 就是非常不錯(cuò)的選擇。
多線程下的單例模式

上面介紹了一些單例模式的基本應(yīng)用方法,但是上面所說的那些使用方式都是有一個(gè)隱含的前提,那就是他們都是應(yīng)用在單線程條件下,一旦換成了多線程就有出錯(cuò)的風(fēng)險(xiǎn)。

如果在多線程的情況下, 餓漢式 不會(huì)出現(xiàn)問題,因?yàn)镴VM只會(huì)加載一次單例類,但是 懶漢式 可能就會(huì)出現(xiàn)重復(fù)創(chuàng)建單例對(duì)象的問題。為什么會(huì)有這樣的問題呢?因?yàn)?懶漢式 在創(chuàng)建單例時(shí)是 線程不安全的,多個(gè)線程可能會(huì)并發(fā)調(diào)用他的 newInstance 方法導(dǎo)致多個(gè)線程可能會(huì)創(chuàng)建多份相同的單例出來。

那有沒有辦法,使 懶漢式 的單利模式也是線程安全的呢?答案肯定是有的,就是使用加同步鎖的方式去實(shí)現(xiàn)。
懶漢式同步鎖

  1. public class Singleton {  
  2.    
  3.     private static Singleton instance = null;  
  4.    
  5.     private Singleton(){  
  6.     }  
  7.    
  8.     public static Singleton getInstance() {  
  9.         synchronized (Singleton.class) {  
  10.             if (instance == null) {  
  11.                 instance = new Singleton();  
  12.             }  
  13.         }  
  14.    
  15.         return instance;  
  16.     }  

這種是最常見的解決同步問題的一種方式,使用同步鎖synchronized (Singleton.class)防止多線程同時(shí)進(jìn)入造成instance被多次實(shí)例化。舉個(gè)在Android使用這種方式的例子:
InputMethodManager示例

  1. public final class InputMethodManager {  
  2.     //內(nèi)部全局唯一實(shí)例    
  3.     static InputMethodManager sInstance;  
  4.      
  5.     //對(duì)外api    
  6.     public static InputMethodManager getInstance() {  
  7.         synchronized (InputMethodManager.class) {  
  8.             if (sInstance == null) {  
  9.                 IBinder b = ServiceManager.getService(Context.INPUT_METHOD_SERVICE);  
  10.                 IInputMethodManager service = IInputMethodManager.Stub.asInterface(b);  
  11.                 sInstance = new InputMethodManager(service, Looper.getMainLooper());  
  12.             }  
  13.             return sInstance;  
  14.         }  
  15.     }  
  16. }   

以上是Android源碼中輸入法類相關(guān)的單例使用方式。

但其實(shí)還有一種更好的方式如下:
雙重校驗(yàn)鎖

  1.  
  2.  
  3. public class Singleton {  
  4.    
  5.     private static volatile Singleton instance = null;  
  6.    
  7.     private Singleton(){  
  8.     }  
  9.    
  10.     public static Singleton getInstance() {  
  11.         // if already inited, no need to get lock everytime  
  12.         if (instance == null) {  
  13.             synchronized (Singleton.class) {  
  14.                 if (instance == null) {  
  15.                     instance = new Singleton();  
  16.                 }  
  17.             }  
  18.         }  
  19.    
  20.         return instance;  
  21.     }  

可以看到上面在synchronized (Singleton.class)外又添加了一層if,這是為了在instance已經(jīng)實(shí)例化后下次進(jìn)入不必執(zhí)行synchronized (Singleton.class)獲取對(duì)象鎖,從而提高性能。

以上兩種方式還是挺麻煩的,我們不禁要問,有沒有更好的實(shí)現(xiàn)方式呢?答案是肯定的。 我們可以利用JVM的類加載機(jī)制去實(shí)現(xiàn)。在很多情況下JVM已經(jīng)為我們提供了同步控制,比如:

在static{}區(qū)塊中初始化的數(shù)據(jù)

訪問final字段時(shí)

等等

因?yàn)樵贘VM進(jìn)行類加載的時(shí)候他會(huì)保證數(shù)據(jù)是同步的,我們可以這樣實(shí)現(xiàn):

采用內(nèi)部類,在這個(gè)內(nèi)部類里面去創(chuàng)建對(duì)象實(shí)例。這樣的話,只要應(yīng)用中不使用內(nèi)部類 JVM 就不會(huì)去加載這個(gè)單例類,也就不會(huì)創(chuàng)建單例對(duì)象,從而實(shí)現(xiàn) 懶漢式 的延遲加載和線程安全。

實(shí)現(xiàn)代碼如下:
靜態(tài)內(nèi)部類

  1.  
  2.  
  3. public class Singleton{  
  4.     //內(nèi)部類,在裝載該內(nèi)部類時(shí)才會(huì)去創(chuàng)建單利對(duì)象  
  5.     private static class SingletonHolder{  
  6.         public static Singleton instance = new Singleton();  
  7.     }  
  8.  
  9.     private Singleton(){}  
  10.  
  11.     public static Singleton newInstance(){  
  12.         return SingletonHolder.instance;  
  13.     }  
  14.  
  15.     public void doSomething(){  
  16.         //do something  
  17.     }  

這樣實(shí)現(xiàn)出來的單例類就是線程安全的,而且使用起來很簡潔,麻麻再也不用擔(dān)心我的單例不是單例了。

然而這還不是最簡單的方式, Effective Java 中推薦了一種更簡潔方便的使用方式,就是使用枚舉。
枚舉類型單例模式

  1. public enum Singleton{  
  2.     //定義一個(gè)枚舉的元素,它就是Singleton的一個(gè)實(shí)例  
  3.     instance;  
  4.  
  5.     public void doSomething(){  
  6.         // do something ...  
  7.     }      

使用方法如下:

  1. public static void main(String[] args){  
  2.    Singleton singleton = Singleton.instance;  
  3.    singleton.doSomething();  

默認(rèn)枚舉實(shí)例的創(chuàng)建是線程安全的.(創(chuàng)建枚舉類的單例在JVM層面也是能保證線程安全的), 所以不需要擔(dān)心線程安全的問題,所以理論上枚舉類來實(shí)現(xiàn)單例模式是最簡單的方式。
總結(jié)

一般單例模式包含了5種寫法,分別是餓漢、懶漢、雙重校驗(yàn)鎖、靜態(tài)內(nèi)部類和枚舉。相信看完之后你對(duì)單例模式有了充分的理解了,根據(jù)不同的場景選擇最你最喜歡的一種單例模式吧!

責(zé)任編輯:陳琳 來源: 推酷
相關(guān)推薦

2013-11-26 16:20:26

Android設(shè)計(jì)模式

2022-06-07 08:55:04

Golang單例模式語言

2021-02-01 10:01:58

設(shè)計(jì)模式 Java單例模式

2021-03-02 08:50:31

設(shè)計(jì)單例模式

2015-09-06 11:07:52

C++設(shè)計(jì)模式單例模式

2022-02-06 22:30:36

前端設(shè)計(jì)模式

2024-02-04 12:04:17

2021-09-07 10:44:35

異步單例模式

2015-01-14 13:26:58

AndroidJava單例

2021-08-11 17:22:11

設(shè)計(jì)模式單例

2022-09-29 08:39:37

架構(gòu)

2022-03-29 07:52:07

設(shè)計(jì)模式單例設(shè)計(jì)模式java

2024-03-06 13:19:19

工廠模式Python函數(shù)

2021-02-07 23:58:10

單例模式對(duì)象

2011-03-16 10:13:31

java單例模式

2023-03-21 15:21:52

開發(fā)程序設(shè)計(jì)static

2020-08-21 07:23:50

工廠模式設(shè)計(jì)

2023-08-03 08:01:27

單例模式結(jié)構(gòu)開發(fā)

2023-07-31 12:27:30

單例設(shè)計(jì)模式

2024-12-09 06:00:00

單例模式代碼
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)