Java多執行緒-----單例模式在多執行緒中的使用用問題
阿新 • • 發佈:2018-12-30
1.餓漢模式(立即載入模式)與多執行緒
不管需不需要用到例項都要去建立例項,即在類產生的時候就建立好例項
package com.thread; /** * 餓漢模式 * * @author yyx 2018年12月28日 */ public class EhanSingleton { public EhanSingleton() { } private static EhanSingleton ehanSingleton = new EhanSingleton(); public staticEhanSingleton getInstance() { try { Thread.sleep(3000); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } return ehanSingleton; } }
package com.thread; public class MyThread extendsThread { public static void main(String[] args) { MyThread m1 = new MyThread(); MyThread m12 = new MyThread(); MyThread m13 = new MyThread(); m1.start(); m12.start(); m13.start(); } @Override public void run() { System.out.println(EhanSingleton.getInstance().hashCode()); } }
執行結果的一種:
507247953
507247953
507247953
在多執行緒下,算出來的結果每個物件的 hashcode的值是一樣的,是執行緒安全的
2.懶漢模式(延遲載入模式)與多執行緒
需要用到建立例項了程式再去建立例項,不需要建立例項程式就“懶得”去建立例項
package com.thread; /** * 懶漢模式 * * @author yyx 2018年12月28日 */ public class LhanSingleton { public LhanSingleton() { } private static LhanSingleton lhanSingleton; public static LhanSingleton getInstance() { if (lhanSingleton == null) { try { Thread.sleep(3000); lhanSingleton = new LhanSingleton(); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } } return lhanSingleton; } }
package com.thread; public class MyThread extends Thread { public static void main(String[] args) { MyThread m1 = new MyThread(); MyThread m12 = new MyThread(); MyThread m13 = new MyThread(); m1.start(); m12.start(); m13.start(); } @Override public void run() { System.out.println(LhanSingleton.getInstance().hashCode()); } }
執行結果的一種:
1486308634
1560533534
507247953
懶漢模式在單執行緒中是符合單例模式的,不過在多執行緒環境中是不符合單例模式
3.懶漢模式執行緒安全的解決機制
3.1 同步程式碼塊
package com.thread; /** * 懶漢模式 * * @author yyx 2018年12月28日 */ public class LhanSingleton { public LhanSingleton() { } private static LhanSingleton lhanSingleton; public static LhanSingleton getInstance() { synchronized (LhanSingleton.class) { if (lhanSingleton == null) { try { Thread.sleep(3000); lhanSingleton = new LhanSingleton(); } catch (InterruptedException e) { e.printStackTrace(); } } } return lhanSingleton; } }
3.2 同步方法
package com.thread; /** * 懶漢模式,不推薦 * * @author yyx 2018年12月28日 */ public class LhanSingleton { public LhanSingleton() { } private static LhanSingleton lhanSingleton; public static synchronized LhanSingleton getInstance() { if (lhanSingleton == null) { try { Thread.sleep(3000); lhanSingleton = new LhanSingleton(); } catch (InterruptedException e) { e.printStackTrace(); } } return lhanSingleton; } }