国产片侵犯亲女视频播放_亚洲精品二区_在线免费国产视频_欧美精品一区二区三区在线_少妇久久久_在线观看av不卡

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - Java教程 - 你真的了解java單例模式了嗎?

你真的了解java單例模式了嗎?

2019-07-04 16:54阿豪聊干貨 Java教程

這篇文章主要介紹了你真的了解java單例模式了嗎?實際上單例模式有著好幾個變種,并且多線程中涉及到線程安全問題,,需要的朋友可以參考下

一、背景

最近在學習設計模式,在看到單例模式的時候,我一開始以為直接很了解單例模式了,實現起來也很簡單,但是實際上單例模式有著好幾個變種,并且多線程中涉及到線程安全問題,那么本文我們就來好好聊聊單例模式,說一下經典三種實現方式:餓漢式、懶漢式、登記式。并且解決掉多線程中可能出現的線程安全問題。

二、基本概念

1.為什么要使用單例模式?

在我們日常的工作中,很多對象通常占用非常重要的系統資源,比如:IO處理,數據庫操作等,那我們必須要限制這些對象只有且始終使用一個公用的實例,即單例。

2.單例模式的實現方式

構造函數私有化,防止其他類生成唯一公用實例外的實例。且單例類應該被定義為final,也就是說單例類不能被繼承,因為如果允許繼承那子類就都可以創建實例,違背了類唯一實例的初衷。

類中一個靜態變量來保存單實例的引用。

一個共有的靜態方法來獲取單實例的引用。
3.單例模式的UML類圖

你真的了解java單例模式了嗎?

4.單例模式的經典實現方式

  • 餓漢式:一開始就創建好實例,每次調用直接返回,經典的“拿空間換時間”。
  • 懶漢式:延遲加載,第一次調用的時候才加載,然后返回,以后的每次的調用就直接返回。經典“拿時間換空間”,多線程環境下要注意解決線程安全的問題。
  • 登記式:對一組單例模式進行的維護,主要是在數量上的擴展,通過線程安全的map把單例存進去,這樣在調用時,先判斷該單例是否已經創建,是的話直接返回,不是的話創建一個登記到map中,再返回。

三、餓漢式---代碼實現

1.單例類

package com.hafiz.designPattern.singleton;
/**
* Desc: 單例模式-餓漢式
* Created by hafiz.zhang on 2017/9/26.
*/
public class Singleton1 {
// 創建全局靜態變量,保證只有一個實例
private static volatile Singleton1 instance = new Singleton1();
private Singleton1() {
// 構造函數私有化
System.out.println("--調用餓漢式單例模式的構造函數--");
}
public static Singleton1 getInstance() {
System.out.println("--調用餓漢式單例模式的靜態方法返回實例--");
return instance;
}
}

2.測試類

public class DesignPatternTest {
@Test
public void testSingleton1() {
System.out.println("-----------------測試餓漢式單例模式開始--------------");
Singleton1 instance1 = Singleton1.getInstance();
System.out.println("第二次獲取實例");
Singleton1 instance2 = Singleton1.getInstance();
System.out.println("instance1和instance2是否為同一實例?" + (instance1 == instance2));
System.out.println("-----------------測試餓漢式單例模式結束--------------");
}
}

3.測試結果

你真的了解java單例模式了嗎?

四、懶漢式---代碼實現

1.單例類

package com.hafiz.designPattern.singleton;
/**
* Desc:單例模式-懶漢式
* Created by hafiz.zhang on 2017/9/26.
*/
public class Singleton2 {
// 創建全局靜態變量,保證只有一個實例
private static Singleton2 instance = null;
// 構造函數私有化
private Singleton2() {
System.out.println("--調用懶漢式單例模式的構造方法--");
}
public static Singleton2 getInstance() {
System.out.println("--調用懶漢式單例模式獲取實例--");
if (instance == null) {
System.out.println("--懶漢式單例實例未創建,先創建再返回--");
instance = new Singleton2();
}
return instance;
}
}

2.測試類

public class DesignPatternTest {
@Test
public void testSingleton2() {
System.out.println("-----------------測試懶漢式單例模式開始--------------");
Singleton2 instance1 = Singleton2.getInstance();
System.out.println("第二次獲取實例");
Singleton2 instance2 = Singleton2.getInstance();
System.out.println("instance1和instance2是否為同一實例?" + (instance1 == instance2));
System.out.println("-----------------測試懶漢式單例模式結束--------------");
}
}

3.測試結果

你真的了解java單例模式了嗎?

細心的同學已經發現,這種實現方式,在多線程的環境中,是有線程安全安全問題的,有可能兩個或多個線程判斷instance都為null,然后創建了好幾遍實例,不符合單例的思想,我們可以對它進行改進。

五、改進懶漢式1---代碼實現

原理:使用JDK的synchronized同步代碼塊來解決懶漢式線程安全問題。

1.單例類

package com.hafiz.designPattern.singleton;
/**
* Desc:單例模式-懶漢式
* Created by hafiz.zhang on 2017/9/26.
*/
public class Singleton2 {
// 創建全局靜態變量,保證只有一個實例
private static Singleton2 instance = null;
// 構造函數私有化
private Singleton2() {
System.out.println("--調用懶漢式單例模式的構造方法--");
}

public static Singleton2 getInstance() {
System.out.println("--調用懶漢式單例模式獲取實例--");
     if (instance != null) {
        System.out.println("--懶漢式單例實例已經創建,直接返回--");
 return instance;
     }
synchronized (Singleton2.class) {
   if (instance == null) {
  System.out.println("--懶漢式單例實例未創建,先創建再返回--");
  instance = new Singleton2();
   }
}
return instance;
}
} 

2.測試結果

你真的了解java單例模式了嗎?

六、改進懶漢式2---代碼實現

原理:使用JVM隱含的同步和類級內部類來解決,JVM隱含的同步解決了多線程情況下線程安全的問題,類級內部類解決只有使用的時候才加載(延遲加載)的問題。

1.JVM隱含的同步有哪些?

靜態初始化器(在靜態字段上或static{}靜態代碼塊的初始化器)初始化數據時

訪問final字段時

在創建線程之前創建對象時

線程可以看見它將要處理的對象時

2.什么是類級內部類?

有static修飾的成員式內部類。沒有static修飾的成員式內部類叫對象級內部類。

類級內部類相當于其外部類的static成分,他的對象與外部類對象間不存在依賴關系,因此可直接創建,而對象級內部類的實例,是綁定在外部對象實例中的。

類級內部類中,可以定義靜態的方法。在靜態的方法中只能夠引用外部類的中的靜態成員方法或者成員變量

類級內部類相當于其外部類的成員,只有在第一次被使用的時候才會被裝載

3.單例類

package com.hafiz.designPattern.singleton;
/**
* Desc:單例模式-改進懶漢式
* Created by hafiz.zhang on 2017/9/26.
*/
public class Singleton3 {
private static class Singleton4 {
private static Singleton3 instance;
static {
System.out.println("--類級內部類被加載--");
instance = new Singleton3();
}
private Singleton4() {
System.out.println("--調用類級內部類的構造函數--");
}
}
private Singleton3() {
System.out.println("--調用構造函數--");
}
public static Singleton3 getInstance() {
System.out.println("--開始調用共有方法返回實例--");
Singleton3 instance;
System.out.println("---------------------------");
instance = Singleton4.instance;
System.out.println("返回單例");
return instance;
}
}

4.測試類

package com.hafiz.www;
import com.hafiz.designPattern.observer.ConcreteObserver;
import com.hafiz.designPattern.observer.ConcreteSubject;
import com.hafiz.designPattern.singleton.Singleton1;
import com.hafiz.designPattern.singleton.Singleton2;
import com.hafiz.designPattern.singleton.Singleton3;
import com.hafiz.designPattern.singleton.Singleton4;
import com.hafiz.designPattern.singleton.Singleton4Child1;
import com.hafiz.designPattern.singleton.SingletonChild2;
import org.junit.Test;
/**
* Desc:設計模式demo單元測試類
* Created by hafiz.zhang on 2017/7/27.
*/
public class DesignPatternTest {
@Test
public void testSingleton3() {
System.out.println("-----------------測試改進懶漢式單例模式開始--------------");
Singleton3 instance1 = Singleton3.getInstance();
System.out.println("第二次獲取實例");
Singleton3 instance2 = Singleton3.getInstance();
System.out.println("instance1和instance2是否為同一實例?" + (instance1 == instance2));
System.out.println("-----------------測試改進懶漢式單例模式結束--------------");
}
}

5.測試結果

你真的了解java單例模式了嗎?

七、登記式--代碼實現

1.基類

package com.hafiz.designPattern.singleton;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
/**
* Desc: 單例模式-登記式
* Created by hafiz.zhang on 2017/9/26.
*/
public class Singleton4 {
private static Map<String, Singleton4> map = new ConcurrentHashMap<>();
protected Singleton4() {
System.out.println("--私有化構造函數被調用--");
}
public static Singleton4 getInstance(String name) {
if (name == null) {
name = Singleton4.class.getName();
System.out.println("--name為空,默認賦值為:--" + Singleton4.class.getName());
}
if (map.get(name) != null) {
System.out.println("name對應的值存在,直接返回");
return map.get(name);
}
System.out.println("name對應的值不存在,先創建,再返回");
try {
Singleton4 result = (Singleton4)Class.forName(name).newInstance();
map.put(name, result);
return result;
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
return null;
}
public Map<String, Singleton4> getMap() {
return map;
}
}

2.子類1

package com.hafiz.designPattern.singleton;
/**
* Desc:
* Created by hafiz.zhang on 2017/9/26.
*/
public class Singleton4Child1 extends Singleton4 {

public static Singleton4Child1 getInstance() {
return (Singleton4Child1) Singleton4.getInstance("com.hafiz.designPattern.singleton.Singleton4Child1");
}
}

3.子類2

package com.hafiz.designPattern.singleton;
/**
* Desc:
* Created by hafiz.zhang on 2017/9/26.
*/
public class SingletonChild2 extends Singleton4 {
public static SingletonChild2 getInstance() {
return (SingletonChild2) Singleton4.getInstance("com.hafiz.designPattern.singleton.SingletonChild2");
}
}

4.測試類

public class DesignPatternTest {
@Test
public void testSingleton4() {
System.out.println("-----------------測試登記式單例模式開始--------------");
System.out.println("第一次取得實例");
Singleton4 instance1 = Singleton4.getInstance(null);
System.out.println("res:" + instance1);
System.out.println("第二次獲取實例");
Singleton4Child1 instance2 = Singleton4Child1.getInstance();
System.out.println("res:" + instance2);
System.out.println("第三次獲取實例");
SingletonChild2 instance3 = SingletonChild2.getInstance();
System.out.println("res:" + instance3);
System.out.println("第四次獲取實例");
SingletonChild2 instance4 = new SingletonChild2();
System.out.println("res:" + instance4);
System.out.println("輸出父類Map中所有的單例");
Map<String, Singleton4> map = instance1.getMap();
for (Map.Entry<String, Singleton4> item : map.entrySet()) {
System.out.println("map-item:" + item.getKey() + "=" + item.getValue());
}
System.out.println("instance1和instance2是否為同一實例?" + (instance1 == instance2));
System.out.println("-----------------測試登記式單例模式結束--------------");
}
}

5.測試結果

你真的了解java單例模式了嗎?

該解決方案的缺點:基類的構造函數對子類公開了(protected),有好的解決方案的博友可以討論指教~

八、總結

經過本文,我們就搞明白了什么叫單例模式,如何優雅的實現經典的單例模式,如何進行拓展和開發具有線程安全的單例模式。對于我們以后的開發非常有幫助,也讓我們更加了解單例模式。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持服務器之家。

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 日韩欧美高清视频 | 97精品国产一区二区三区 | 精品国产乱码久久久久久丨区2区 | 99精品一区二区 | 色乱码一区二区三区网站 | 99精品久久久久久久免费 | 国产精品九九九 | 91久久久久久 | 久久天天躁狠狠躁夜夜躁2014 | 国产欧美精品一区二区三区 | 国产二区视频 | 中文字幕日韩在线视频 | 日韩一区在线视频 | 做a视频免费观看 | 51ⅴ精品国产91久久久久久 | 免费看一区二区三区 | 18视频在线观看网站 | 色综合天天综合网国产成人网 | 精品成人久久 | 亚洲xxxx3d| 久久影院免费观看 | 欧美不卡视频 | 一色视频 | 欧美激情一区二区 | 久久亚洲欧美日韩精品专区 | 亚洲乱码国产乱码精品精的特点 | 久久视频精品 | 日韩福利视频 | 日韩欧美国产一区二区 | 日韩在线免费 | 日本免费在线视频 | 久草新免费 | 女人久久久久 | 五月婷婷在线观看视频 | 精品久久一区二区三区 | 成人精品视频在线观看 | 亚洲第一视频 | 男女羞羞网站 | 久久精品高清 | 久热中文在线 | 91 在线免费观看 |