Singleton和Double-Checked Locking设计模式,分别指的是单例模式和双重检查锁模式,它们都可以用于确保某个类只有一个对象实例化。

两个模式的区别在于:Singleton模式用在单线程应用程序中,而Double-Checked Locking模式用于多线程模式。

一、Singleton模式

UML图:

代码:

  1. package bupt.xujinliang.singletonpattern;
  2. /**
  3. *
  4. * @author jin
  5. *
  6. */
  7. public class SingletonExample {
  8. public static void main(String[] args) {
  9. Printer printer1 = Printer.getInstance();
  10. Printer printer2 = Printer.getInstance();
  11. if(printer1 == printer2) {
  12. System.out.println("printer2 point to the same address with printer1");
  13. } else {
  14. System.out.println("printer2 point to different address with printer1");
  15. }
  16. }
  17. }
  18. class Printer {
  19. private static Printer instance;
  20. public Printer() {
  21. System.out.println("Printer Constructor");
  22. }
  23. public static Printer getInstance() {
  24. if(null == instance)
  25. instance = new Printer();
  26. return instance;
  27. }
  28. }

运行结果:

2.Double-Checked Locking模式

Double Check Locking模式是singleton的多线程版本,必须使用锁来锁定临界区,当多个线程存在访问临界区的意图时,保证了临界区只被访问一次。
首先介绍其在C/C++环境下的实现过程:
代码1:
  1. Printer* get_instance(void)
  2. {
  3. lock();
  4. if( instance == 0) {
  5. instance = new Printer;
  6. }
  7. unlock();
  8. return instance;
  9. }

上述代码存在的问题是:无论是否已经初始化都要加锁,增加了负荷,已经没有所谓的并发性能了。

代码2:
  1. Printer* get_instance(void)
  2. {
  3. if( instance == 0){
  4. lock();
  5. instance = new Printer;
  6. unlock();
  7. }
  8. return instance;
  9. }

上述代码存在的问题是:不能保证临界区只初始化一次,没能实现singleton的基本功能。

代码3:
  1. Printer* get_instance(void)
  2. {
  3. if( instance == 0){
  4. lock();
  5. if( instance == 0 )
  6. instance = new Printer;
  7. unlock();
  8. }
  9. return instance;
  10. }

这是比较完善的Double-Checked Locking模式实现的代码。

为什么叫做Double-Checked Locking呢?请看上述代码3,可以看到在加锁前后都对instance变量进行了检查,故谓之Double-Checked Locking。
那么在Java中的实现与在C/C++中不同吗?是的。
下面的的Java代码是不能够实现Double-Checked Locking模式的:
  1. class Printer {
  2. private static Printer resource ;
  3. public static Printer getInstance(){
  4. if(resource == null ){
  5. synchronized (DoubleCheckedLockingExample.class) {
  6. if(resource  == null ){
  7. resource  = new Printer() ;
  8. }
  9. }
  10. }
  11. return resource ;
  12. }
  13. private Printer(){}
  14. }

上面程序真正的问题是没有同步的情况下读取共享变量resource,并发的情况下对象的状态值有可能是过期无效的。要解决这个问题也很简单,把resource声明为volatile类型。volatile有什么作用?引用《java并发编程实战》的解析:

  1. 当一个域声明为volatile类型后,编译器与运行时会监视这个变量:它是共享的,而且对它的操作不会与其他的内存操作一起被重排序。volatile变量不会缓存在寄存器或缓存在对其他处理器隐藏的地方。所以,读一个volatile类型的变量时,总会返回由某一线程所写入的最新值。

读取volatile变量比读取非volatile变量的性能几乎没有差别,不过需要注意的是volatile只能保证内存可见性,并不能保证原子性。

现给出Java在多线程下实现单个实例化对象的方法:
  1. class Printer {
  2. private static class Instance {
  3. static final Printer instance = new Printer();
  4. }
  5. private static Printer resource ;
  6. public static Printer getInstance(){
  7. return Instance.instance;
  8. }
  9. private Printer(){}
  10. }

上述方法之所以有效,是因为内部类(Instance)将只被装载一次,所以只会创建一个对象。

[zt]Singleton和Double-Checked Locking设计模式—UML图及代码实现的更多相关文章

  1. Java中的双重检查锁(double checked locking)

    最初的代码 在最近的项目中,写出了这样的一段代码 private static SomeClass instance; public SomeClass getInstance() { if (nul ...

  2. Double Checked Locking 模式

    转自:http://blog.csdn.net/wwsoon/article/details/1485886 之前在使用Double Check Locking 模式时,发现自己还是不太理解.于是写个 ...

  3. 设计模式-UML图简单介绍

    直接上法宝: 1.类(Class)     类图分三层:     第一层显示类的名称,如果是抽象类,则就用斜体显示.     第二层是类的特性,通常就是字段和属性.     第三层是类的操作,通常是方 ...

  4. 设计模式UML图

    1.简单工厂模式 2.工厂模式 工厂模式与简单工厂模式的不同在于,每个操作类都有自己的工厂,而且把逻辑判断交给了客户端,而简单工厂的逻辑判断在工厂类里边,当增加新的操作类时,简单工厂需要修改工厂类,而 ...

  5. 【转】23种设计模式UML图

    原文:http://blog.csdn.net/bwwlpnn/article/details/7421628

  6. 三类设计模式UML图

    http://design-patterns.readthedocs.org/zh_CN/latest/index.html

  7. 23种设计模式UML图

  8. 设计模式 UML & java code

    A: 创造性模式 1. 工厂方法模式(FactoryMethod) 1.1 类图 1.2 代码1 public interface Pet { public String petSound(); } ...

  9. 23种经典设计模式UML类图汇总

    在这里23种经典设计模式UML类图汇总       创建型模式 1.FACTORY—追MM少不了请吃饭了,麦当劳的鸡翅和肯德基的鸡翅都是MM爱吃的东西,虽然口味有所不同,但不管你带MM去麦当劳或肯德基 ...

随机推荐

  1. MFC GDI绘图基础

    一.关于GDI的基本概念 什么是GDI? Windows绘图的实质就是利用Windows提供的图形设备接口GDI(Graphics Device Interface)将图形绘制在显示器上. 在Wind ...

  2. felx项目属性(二)

    order flex-grow flex-shrink flex-basis flex align-self 1.1 order css order属性规定了弹性容器中的可伸缩项目在布局时的顺序.元素 ...

  3. ArrayList集合&特殊集合

    一.ArrayList集合 集合内可以放不同类型的元素 另:object类型为所有数据类型的基类 添加元素:.add(); 清空集合:al.clear(); 克隆集合:.clone(); 判断是否包含 ...

  4. SQLServer 维护脚本分享(09)相关文件读取

    /********************[读取跟踪文件(trc)]********************/ --查看事件类型描述 SELECT tc.name,te.trace_event_id, ...

  5. nginx日志中文变成类似\xE9\xA6\x96\xE9\xA1\xB5-\xE6\x8E\xA8\xE8\x8D\x90的东西,治本方案

    这里:https://groups.google.com/forum/#!topic/openresty/NcRSb5gTmVU 主要是: 这与 ngx_lua 无关,是较新的 nginx 核心引入的 ...

  6. 分享一个漂亮WPF界面框架创作过程及其源码(转)

    本文会作为一个系列,分为以下部分来介绍: (1)见识一下这个界面框架: (2)界面框架如何进行开发: (3)辅助开发支持:Demo.模板.VsPackage制作. 框架源码如下所示. 本文介绍第(1) ...

  7. 实现Web验证码图片-原理

    实现验证码的基础 GDI+ graphics device interface plus的缩写,即图形设备接口.GDI+为开发者提供了一组实现与各种设备(具有图形化能力但不涉及图形细节的设备)进行交互 ...

  8. Codeforces Round #375 (Div. 2) - C

    题目链接:http://codeforces.com/contest/723/problem/C 题意:给定长度为n的一个序列.还有一个m.现在可以改变序列的一些数.使得序列里面数字[1,m]出现次数 ...

  9. BZOJ 2342 回文串-Manacher

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2342 思路:先跑一遍Manacher求出p[i]为每个位置为中心的回文半径,因为双倍回文串 ...

  10. js不间断平滑地自动向上滚动

    <html> <head> <title>scroll up auto smooth</title> <style> *{ margin: ...