一篇文章讓你徹底搞懂單例設計模式

来源:https://www.cnblogs.com/java265/archive/2022/05/12/16264765.html
-Advertisement-
Play Games

前言 **基礎篇鏈接:**https://www.cnblogs.com/xiegongzi/p/16229678.html 3.9、延遲隊列 - 重要 3.9.1、延遲隊列概念 這個玩意兒要表達的意思其實已經見過了,就是死信隊列中說的TTL消息過期,但是文字表達得換一下 所謂的延遲隊列:就是用來存 ...


轉自:

http://www.java265.com/JavaCourse/202109/1153.html

下文是筆者編寫的單例模式實現的八種方式,如下所示:

單例模式的簡介

我們將一個類在當前進程中只有一個實例的這種模式,稱之為“單例模式”
那麼Java代碼如何實現一個單例模式呢?下文將一一到來,如下所示:

單例模式的註意事項:
   1.單例模式在一個進程中只有一個實例
   2.單例類通常由自己創建自身的實例
   3.單例類給其他對象提供的都是同一個實例

測試代碼

package com.java265.Singleton;

public class Test {

    public static void main(String[] args) {
        // TODO Auto-generated method stub

        System.out.println("------單例模式-----");
        
        //創建100個線程進行測試
        
        for(int i=0;i<100;i++) {
            new Thread(()->{
                System.out.println(Single01.getInstance());
             }).start();
        }
    }

}

 

單例實現模式1

餓漢式單例模式:
直接使用一個靜態變數,在JVM載入類時,生成一個單例實例 如下

package com.java265.Singleton;

public class Single01 {    
    private static final Single01 INSTANCE = new Single01();

    private Single01() {}
    
    public static  Single01 getInstance () {
        return INSTANCE;
    }
}

 

使用static靜態代碼塊生成一個單例類

package com.java265.Singleton;
public class Single02 {
    private static final Single02  INSTANCE;
    
    static {
         INSTANCE = new Single02();
    }
       
    private Single02() {}

    public static Single02 getInstance() {
        return INSTANCE;    
    }
    public void t() {
         System.out.println("Single02 t方法"
                 + "");
    }
}

 

使用判斷的方式,創建單例模式,
但是此處不是一個線程安全的創建方式

package com.java265.Singleton;

/*
 * 這是一個線程不安全的創建單例模式的方式
 * 這是一個懶漢式的創建單例模式的方式
 * */
public class Single03 {
    private static Single03 INSTANCE;
    private Single03() {    
    }

    public  static Single03 getInstance() {
         if(INSTANCE ==null)
         {
                // 多個線程都會被卡在此處,
                // 當sleep運行完畢後,多個線程會同時創建實例,此處的代碼是產生線程不安全的根源
             try
             {
              Thread.sleep(100);
             }catch(InterruptedException e)
             {
                 
             }
             INSTANCE = new Single03();
         }
         
         return INSTANCE;
    }    
}

 

使用 synchronized為方法加上鎖,使其線程安全

package com.java265.Singleton;
public class Single04 {
    private static volatile Single04 INSTANCE;
    private Single04() {    
    }

    /*
     * 懶漢式生成單例實例 此處使用 synchronized 安全鎖
     */
    public  static  synchronized Single04 getInstance() {
         if(INSTANCE ==null)
         {
             try
             {
              Thread.sleep(100);
             }catch(InterruptedException e)
             {
                 
             }
             INSTANCE = new Single04();
         }
         
         return INSTANCE;
    }    
}

 

減少鎖粒度,將synchronized關鍵字直接加在方法內部具體的位置上

package com.java265.Singleton;
public class Single05 {

    private static Single05 INSTANCE;
    private Single05() {
        
    }
    public  static   Single05 getInstance() {
         if(INSTANCE ==null)
         {
                /*
                 * 將鎖直接加到方法體裡面 此時出現了一個新的問題 當所有的線程都堵塞在此處,也會創建多個實例
                 */
             synchronized(Single05.class) {
             try
             {
              Thread.sleep(100);
             }catch(InterruptedException e)
             {
                 
             }
             INSTANCE = new Single05();
             }
         }     
         return INSTANCE;
    }    
}

 

將synchronized鎖放入在方法體中,同時使用雙重檢查,避免創建多個實例

package com.java265.Singleton;

public class Single06 {

    private static Single06 INSTANCE;
    private Single06() {
        
    }

    public  static   Single06 getInstance() {
         if(INSTANCE ==null)
         {
             synchronized(Single06.class) {
             try
             {
              Thread.sleep(100);
             }catch(InterruptedException e)
             {
                 
             }
             if(INSTANCE == null)
             {
             INSTANCE = new Single06();
             }
             }
         }
         
         return INSTANCE;
    }    
}

 

使用靜態內部類的方式創建一個單例對象
此方式主要藉助JVM載入類時,內部類不會被載入
當我們使用內部類的時,才會被載入,此時由JVM保證靜態內部類的唯一性

package com.java265.Singleton;

/*
 * 採用內部類的方式實現一個單例模式
 * */
public class Single07 {

    private Single07() {
        
    }

    private static class Single07Holder {
        private final static Single07 INSTANCE = new Single07();
    }

    public  static   Single07 getInstance() {
        return Single07Holder.INSTANCE;
    }    
}

 

使用枚舉創建一個靜態內部類

package com.java265.Singleton;

/*
 * 採用枚舉實現一個單例模式
 * */
public enum Single08 {

    INSTANCE;

    public static Single08 getInstance() {
        return INSTANCE;
    }     
}

 


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

-Advertisement-
Play Games
更多相關文章
  • 本文介紹什麼是 SQL 的聚集函數,如何利用它們彙總表的數據。這些函數很高效,它們返回結果一般比你在自己的客戶端應用程式中計算要快得多。 一、聚集函數 我們經常需要彙總數據而不用把它們實際檢索出來,為此 SQL 提供了專門的函數。使用這些函數,SQL 查詢可用於檢索數據,以便分析和報表生成。這種類型 ...
  • 本文介紹什麼是函數,DBMS 支持何種函數,以及如何使用這些函數;還將講解為什麼 SQL 函數的使用可能會帶來問題。 一、函數 與大多數其他電腦語言一樣,SQL 也可以用函數來處理數據。函數一般是在數據上執行的,為數據的轉換和處理提供了方便。 SQL 如何創建計算欄位 中用來去掉字元串尾的空格的 ...
  • 本文介紹什麼是計算欄位,如何創建計算欄位,我們用例子說明瞭計算欄位在字元串拼接和算術計算中的用途。以及如何從應用程式中使用別名引用它們。 一、計算欄位 存儲在資料庫表中的數據一般不是應用程式所需要的格式,下麵舉幾個例子。 需要顯示公司名,同時還需要顯示公司的地址,但這兩個信息存儲在不同的表列中。 城 ...
  • 轉載請標明出處,維權必究: https://www.cnblogs.com/tangZH/p/12543154.html Glide作為一個強大的圖片載入框架,已經被android官方使用,所以,明白Glide的載入流程以及原理對加深我們對glide的理解是很重要的。 本文基於glide 4.11 ...
  • 在 OpenHarmony 生態發展過程中,涌現了大批優秀的代碼貢獻者,本專題旨在表彰貢獻、分享經驗,文中內容來自嘉賓訪談,不代表 OpenHarmony 工作委員會觀點。 ...
  • Vue3使用插槽時的父子組件傳值 用法見官方文檔深入組件章節,插槽部分: 參考文檔:插槽-作用域插槽-插槽prop 作用域插槽 有時讓插槽內容能夠訪問子組件中才有的數據是很有用的。 需求:插槽內容能夠訪問子組件中才有的數據 實現 子組件 TodoList.vue <template> <div v- ...
  • C++自定義迭代器:介紹了【什麼時候需要用到自定義迭代器】和【如何實現自定義迭代器】。 ...
  • 環境 python 版本3.6.4 gevent 1.5.0 gunicorn 20.1.0 錯誤 RecursionError: maximum recursion depth exceeded while calling a Python object 錯誤原因 根據錯誤棧,出問題的代碼在pyt ...
一周排行
    -Advertisement-
    Play Games
  • Dapr Outbox 是1.12中的功能。 本文只介紹Dapr Outbox 執行流程,Dapr Outbox基本用法請閱讀官方文檔 。本文中appID=order-processor,topic=orders 本文前提知識:熟悉Dapr狀態管理、Dapr發佈訂閱和Outbox 模式。 Outbo ...
  • 引言 在前幾章我們深度講解了單元測試和集成測試的基礎知識,這一章我們來講解一下代碼覆蓋率,代碼覆蓋率是單元測試運行的度量值,覆蓋率通常以百分比表示,用於衡量代碼被測試覆蓋的程度,幫助開發人員評估測試用例的質量和代碼的健壯性。常見的覆蓋率包括語句覆蓋率(Line Coverage)、分支覆蓋率(Bra ...
  • 前言 本文介紹瞭如何使用S7.NET庫實現對西門子PLC DB塊數據的讀寫,記錄了使用電腦模擬,模擬PLC,自至完成測試的詳細流程,並重點介紹了在這個過程中的易錯點,供參考。 用到的軟體: 1.Windows環境下鏈路層網路訪問的行業標準工具(WinPcap_4_1_3.exe)下載鏈接:http ...
  • 從依賴倒置原則(Dependency Inversion Principle, DIP)到控制反轉(Inversion of Control, IoC)再到依賴註入(Dependency Injection, DI)的演進過程,我們可以理解為一種逐步抽象和解耦的設計思想。這種思想在C#等面向對象的編 ...
  • 關於Python中的私有屬性和私有方法 Python對於類的成員沒有嚴格的訪問控制限制,這與其他面相對對象語言有區別。關於私有屬性和私有方法,有如下要點: 1、通常我們約定,兩個下劃線開頭的屬性是私有的(private)。其他為公共的(public); 2、類內部可以訪問私有屬性(方法); 3、類外 ...
  • C++ 訪問說明符 訪問說明符是 C++ 中控制類成員(屬性和方法)可訪問性的關鍵字。它們用於封裝類數據並保護其免受意外修改或濫用。 三種訪問說明符: public:允許從類外部的任何地方訪問成員。 private:僅允許在類內部訪問成員。 protected:允許在類內部及其派生類中訪問成員。 示 ...
  • 寫這個隨筆說一下C++的static_cast和dynamic_cast用在子類與父類的指針轉換時的一些事宜。首先,【static_cast,dynamic_cast】【父類指針,子類指針】,兩兩一組,共有4種組合:用 static_cast 父類轉子類、用 static_cast 子類轉父類、使用 ...
  • /******************************************************************************************************** * * * 設計雙向鏈表的介面 * * * * Copyright (c) 2023-2 ...
  • 相信接觸過spring做開發的小伙伴們一定使用過@ComponentScan註解 @ComponentScan("com.wangm.lifecycle") public class AppConfig { } @ComponentScan指定basePackage,將包下的類按照一定規則註冊成Be ...
  • 操作系統 :CentOS 7.6_x64 opensips版本: 2.4.9 python版本:2.7.5 python作為腳本語言,使用起來很方便,查了下opensips的文檔,支持使用python腳本寫邏輯代碼。今天整理下CentOS7環境下opensips2.4.9的python模塊筆記及使用 ...