GOF設計模式之1:單例設計模式

来源:http://www.cnblogs.com/chun-jiang-chao-de-gu-shi/archive/2016/04/10/5376574.html
-Advertisement-
Play Games

1.單例設計模式核心作用: 保證一個類只有一個實例,並且提供了訪問該實例的全局訪問點 2.常見應用場景: window的任務管理器 項目中讀取配置文件一般也是一個單例模式 資料庫連接池的設計也是採用單例模式,因為資料庫連接是一種資料庫資源 操作系統的文件管理系統,也是單例模式,一個操作系統只能有一個 ...


1.單例設計模式核心作用:

保證一個類只有一個實例,並且提供了訪問該實例的全局訪問點

2.常見應用場景:

  • window的任務管理器
  • 項目中讀取配置文件一般也是一個單例模式
  • 資料庫連接池的設計也是採用單例模式,因為資料庫連接是一種資料庫資源
  • 操作系統的文件管理系統,也是單例模式,一個操作系統只能有一個文件系統
  • Application也是單例的應用(Servlet編程或者Android的Application類)
  • 在Spring中,每個bean預設也是單例的,這樣的有點兒事Spring容器可以管理
  • 在Servlet編程中每個Servlet也是單例的
  • 在Spring MVC和Struts1框架中控制器對象也是單例

3.單例模式的優點

  • 由於單例模式只生產一個對象,減少了系統開銷,當一個對象的產生需要的資源比較多的時候,比如讀取配置文件、產生其它依賴對象時,則可以在應用啟動的時候直接產生一個單例對象,然後永久駐存記憶體的方式來解決。
  • 單例模式可以在系統設置全局訪問點,優化共用資源的訪問。例如可以設計一個單例類,負責所有數據表的映射。

4.常見5中單例模式的實現方式:

主要

餓漢式:線程安全,調用效率高。但是不能延時載入

懶漢式:線程安全,調用效率不高。但是可以延遲載入

其它:

雙重檢鎖式:由於JVM底層內部模型的原因,偶爾會出現問題,不建議使用

靜態內部類式:線程安全,調用效率高,而且可以延遲載入

枚舉單例:線程安全,調用效率高,不可延遲載入

餓漢式的示例代碼:

public class Singleton01 {
    //類初始化的時候,立即載入這個對象(沒有延時載入的優勢)。載入類時,是線程安全的
    private static Singleton01 instance = new Singleton01();
    private Singleton01(){}
    //方法沒有同步調用效率高
    public static Singleton01 getInstance(){
        return instance;
    }
}

 

餓漢式單例模式的代碼中,static變數會在類裝載的時候進行初始化,此時不會涉及到多個線程對象訪問該對象的問題。虛擬機會保證只會裝載一次該類,肯定不會發生併發訪問的問題,因此可以省略synchronized關鍵字

問題:如果僅僅是載入本類,而不是要調用getInstance,甚至永遠都沒有調用,則會造成資源浪費。

懶漢式的示例代碼

 1 package com.chunjiangchao.pattern.singleton;
 2 /**
 3  * 測試懶漢式單例模式
 4  */
 5 public class Singleton02 {
 6     //類初始化的時候,不初始化這個對象(延時載入,真正用的時候再創建)。
 7     private static Singleton02 instance = null;
 8     private Singleton02(){}
 9     ////方法同步,調用效率低!
10     public static synchronized Singleton02 getInstance(){
11         if(instance == null)
12             instance = new Singleton02();
13         return instance;
14     }
15 }

 

要點:延遲載入,懶載入真正用到的時候才會選擇載入

問題:

資源利用率高了,但是每次調用getInstance()方法都要同步,併發效率較低。

雙重檢鎖實現

 1 package com.chunjiangchao.pattern.singleton;
 2 /**
 3  * 測試DCL(雙重檢鎖)單例模式
 4  * 
 5  */
 6 public class Singleton03 {
 7     //類初始化的時候,不初始化這個對象(延時載入,真正用的時候再創建)。
 8     private volatile static Singleton03 instance = null;
 9     private Singleton03(){}
10     ////代碼塊同步,調用效率要比同步方法要快一些,由於JVM的原因在高併發的情況下會出現問題
11     public static  Singleton03 getInstance(){
12         if(instance == null){
13             synchronized (Singleton03.class) {
14                 if(instance == null)
15                     instance = new Singleton03();
16             }
17         }
18         return instance;
19     }
20 }

 

提高了執行 的效率,不必每次獲取對象的時候都要進行同步,只有第一次才會進行同步創建。

問題:

由於編譯器優化的原因和JVM底層內部模型原因,偶爾會出現問題,不建議使用。但是我們可以在instance前面添加volatile關鍵字,這樣就沒問題了。

靜態內部類實現方式:(懶載入方式)

 1 package com.chunjiangchao.pattern.singleton;
 2 /**
 3  * 靜態內部類單例模式
 4  * 這種方式:線程安全,調用效率高,並且實現了延時載入!
 5  */
 6 public class Singleton04 {
 7     private Singleton04(){}
 8     public static  Singleton04 getInstance(){
 9         return Inner.instance;
10     }
11     private static class Inner{
12         private static final Singleton04 instance = new Singleton04();
13     }
14 }

 

外部類沒有static屬性,則不會像餓漢式那樣,上來就把對象造出來了

只有真正調用getInstance才會載入靜態內部類。載入類時是線程安全的。instance 是static final類型,保證了記憶體中只有這樣一個實例存在,而且只被賦值一次,從而保證了線程安全性。

兼併併發高效調用和延遲載入的優勢。

換一句戶說:靜態內部有具備餓漢式和延遲載入的優勢。

枚舉實現單例:

 1 package com.chunjiangchao.pattern.singleton;
 2 /**
 3  * 枚舉式實現單例模式(沒有延時載入)
 4  */
 5 public enum Singleton05 {
 6     instance;//這個枚舉元素,本身就是單例對象!
 7     public void operation(){
 8         //添加需要的操作
 9     }
10 }

 

優點:實現簡單;枚舉本身就是單例。由JVM從根本上提供保障。避免反射和序列化的漏洞

缺點:無延遲載入

5.如何選用這五種單例模式?

單例對象占用資源少,不需要延遲載入:

枚舉好於餓漢式

單例對象占用資源大,需要延遲載入

靜態內部類好於懶漢式

6.問題

反射可以破解上面(不包含枚舉)的實現方式(防止的做法是在構造方法中手動拋出異常)

反序列化可以破解(不包含枚舉)的實現方式

可以通過定義readResolve防止獲得不同對象。反序列化的時候,如果對象所在的類定義了readResolve()方法(一種回調方法),返回自己創建的那個對象。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 基本數據結構:鏈表(list) 談到鏈表之前,先說一下線性表。線性表是最基本、最簡單、也是最常用的一種數據結構。線性表中數據元素之間的關係是一對一的關係,即除了第一個和最後一個數據元素之外,其它數據元素都是首尾相接的。線性表有兩種存儲方式,一種是順序存儲結構,另一種是鏈式存儲結構。 順序存儲結構就是 ...
  • 深入理解和探究Java類載入機制 1.java.lang.ClassLoader類介紹 java.lang.ClassLoader類的基本職責就是根據一個指定的類的名稱,找到或者生成其對應的位元組代碼,然後從這些位元組代碼中定義出一個Java 類,即 java.lang.Class類的一個實例。 Cla ...
  • 1.產生 源碼(.java文件)——>編譯器(如:javac)——>位元組碼(.class文件)——>虛擬機(如:HotSpot)執行 2.Class文件 1)構成: 2)例子: [1] .java文件 桌面/test/test/Father.java、桌面/test/test/Father_inte ...
  • ...
  • 一、PDO的概念 PDO其實就是一個資料庫的抽象層,使用PDO編程可以方便的在之後的實際運營中隨時更改資料庫而不用變更源代碼。PDO的位置如下圖所示: 二、PDO的開啟 PDO需要使用php 5.1 之後的版本。 查看是否開啟pdo功能需要新建一個php文件,使用phpinfo函數查詢 如圖,PDO ...
  • 本文原創作者:pipi-changing本文原創出處:http://www.cnblogs.com/pipi-changing/ 本文版權歸作者和博客園共有,未經作者同意必須保留此段聲明,且在文章頁面明顯位置給出原文連接 ,否則保留追究法律責任的權利。 面向對象概念 理解面向對象 面向對象是相對面向 ...
  • 培訓大數據架構開發! 從零基礎到高級,一對一培訓![技術QQ:2937765541] 課程體系: 獲取視頻資料和培訓解答技術支持地址 課程展示(大數據技術很廣,一直線上為你培訓解答!): 獲取視頻資料和培訓解答技術支持地址 ...
  • 1、整型 取值範圍如果加了unsigned,則最大值翻倍,如tinyint unsigned的取值範圍為(0~256)。 int(m)里的m是表示SELECT查詢結果集中的顯示寬度,並不影響實際的取值範圍,沒有影響到顯示的寬度,不知道這個m有什麼用。 2、浮點型(float和double) 設一個字 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...