`

《Head First设计模式》阅读笔记.第五章

    博客分类:
  • Java
阅读更多
1.单件(单态,Singleton)模式部分

*有些对象我们只需要一个,比如说:线程池(threadpool)、缓存(cache)、对话框()、处理偏好设置的对象、处理注册表(register)的对象、日志对象,以及充当打印机、显卡等设备的驱动程序对象。这些对象只能有一个实例,如果出现多个实例就会导致程序的行为异常、资源使用过量,或者产生的结果不一致等等问题。

*单件模式与全局静态变量的区别:
(1)使用全局静态变量需要程序员之间的约定才能保证只有一个实例,而单件模式无需这样的约定就可以确保只有一个实例被创建。
(2)静态变量在程序一开始就被创建(这取决于JVM的实现),而单件模式只是在使用时才创建对象。如果这个被创建的对象非常消耗资源,而在程序运行的过程中没有用到它,就会造成很大的浪费,这是静态变量的缺点。

*在单态模式中,如果不需要这个实例,它就永远不会被创建。这就是“延迟实例化(Lazy Instance)”。

*单件模式的应用场景之一是设置类对象,比如注册表设置(Register Setting)对象。如果设置对象有多份拷贝,就会把设置搞得一团糟。

*单件常用来管理共享的资源,比如数据库连接池或线程池。

单件(Singleton)模式:确保一个类只有一个实例,并提供一个全局访问点。

*多线程会影响到单件模式,如果不对它进行处理就会在单件模式下仍然创建多于一个实例。
解决这个问题有以下三种方式:
(1)使用同步。但是简单地给创建实例方法(getInstance())增加synchronized修饰符虽然可以解决多线程的问题,但是导致每次调用都同步,而在静态变量被设置之后,同步就是多余的了,因此,这降低了程序的效率。
在程序频繁运行的地方增加同步可能会使效率降低100倍!因此要尽量避免使用同步,如果使用,就要尽量缩减需要同步的代码。
方法如下:
------------
public class Singleton {
	private static Singleton instance;
	
	private Singleton() {}
	
	public synchronized static Singleton getInstance() {
	    if (instance == null) {
	        instance = new Singleton();
	    }
	    return instance;
	}
}
------------

(2)使用“急切(eagerly)”创建实例,也就是在生命静态变量的时候就创建实例,而不是等到使用的时候再创建。该方式适用于程序总是需要创建和使用单件实例、程序在创建和运行时负担不是太重、单件实例占用的资源较少的情况。
方法如下:
------------
public class Singleton {
    private static Singleton instance = new Singleton();
    
    private Singleton() {}
    
    public static Singleton getInstance() {
        return instance;
    }
}
------------

(3)在创建实例方法(getInstance())方法中使用“双重检查加锁(Double-Checked Locking)”,这样既保持了“延迟实例化(Lazy Instance)”,又保证只在第一次调用时同步。
方法如下:
------------
public class Singleton {
	private volatile static Singleton instance;
	
	private Singleton() {}
	
	public static Singleton getInstance() {
	    if (instance == null) {
	        synchronized (Singleton.class) {
	            if (instance == null) {
	                instance = new Singleton();
	            }
	        }
	    }
	    return instance;
	}
}
------------

在这个方法里使用到了volatile这个关键字,下面对这个“关键的”关键字进行说明:
Java语言规范指出,为了获得最佳速度,允许线程保存共享成员变量的私有拷贝,而且只当线程进入和离开同步代码块时才与共享成员变量的原始值进行对比。
而被volatile修饰的成员变量在线程中的私有拷贝每次被线程访问时,都强迫从共享内存中重读该成员变量的值。而且,当成员变量的私有拷贝发生变化时,强迫线程将变化值回写到共享内存。这样在任何时刻,两个不同的线程总是看到某个成员变量的同一个值。
因此volatile关键字是使“双重检查加锁(Double-Checked Locking)”有效的前提。
但是需要注意,在1.4及以前版本的JDK中,JVM对volatile关键字的实现会导致双重检查加锁失效,所以这个方法只适用于1.5(包含)以后版本。

*以上对三种处理多线程方法的总结也就是“Sharpen Your Pencil”的解答。

*可以通过把一个类中的全部方法都定义为静态方法的方式来达到和单件模式同样的效果,但是由于类的初始化顺序由JVM控制,所以可能导致与初始化顺序有关的BUG,而这样的BUG常常难于被发现。当类的初始化比较简单时,可以使用此方法。

*类加载器会破坏单件模式,因为不同的类加载器可以分别创建同一个单件的对象,单件对象就有了多个实例。解决办法是:自行指定类加载器,并且指定同一个类加载器。

*在Java 1.2及以前版本中,垃圾收集器有一个BUG,会造成单件在没有全局引用时,被当做垃圾清理掉。在1.2以后的版本中,这个BUG已经得到了修复,因此不用担心了。
如果使用的是1.2及以前的版本,可以建立一个单件注册表(Register),从而避免单件被垃圾收集器回收。

*虽然单件模式不支持继承,但在一个软件中用到它的机会并不多,所以这个限制几乎没有影响。

*Java中实现单件(Singleton)模式需要私有的构造器、一个静态变量和一个静态方法。

2.单件(Singleton)模式实例

第一种实现:
public class ThreadPool {
	private static ThreadPool instance;

	private ThreadPool() {
	}

	public synchronized static ThreadPool getInstance() {
		if (instance == null) {
			instance = new ThreadPool();
		}
		return instance;
	}
}


第二种实现:
public class DBConnectionPool {
	private static DBConnectionPool instance = new DBConnectionPool();

	private DBConnectionPool() {
	}

	public static DBConnectionPool getInstance() {
		return instance;
	}
}


第三种实现(适用于1.5及以后版本):
public class LogFactory {
	private volatile static LogFactory instance;

	private LogFactory() {
	}

	public static LogFactory getInstance() {
		if (instance == null) {
			synchronized (LogFactory.class) {
				if (instance == null) {
					instance = new LogFactory();
				}
			}
		}
		return instance;
	}
}


--END--
4
0
分享到:
评论

相关推荐

    笔记_HeadFirst设计模式.pdf

    笔记_HeadFirst设计模式

    Head.First 设计模式学习笔记.pdf

    ### Head.First 设计模式学习笔记知识点总结 #### 一、设计模式概述 设计模式是一种用于解决软件设计中常见问题的标准化方法。通过采用设计模式,开发者可以提高代码的复用性、灵活性和可维护性。《Head First 设计...

    HeadFirst设计模式学习笔记

    《HeadFirst设计模式学习笔记》是一份详尽的资料,旨在帮助读者深入理解并掌握设计模式这一编程领域的核心概念。设计模式是软件工程中的一种最佳实践,它在解决常见问题时提供了一种标准的解决方案,使得代码更易于...

    HeadFirst 设计模式学习笔记1--策略模式Demo

    《HeadFirst设计模式学习笔记1--策略模式Demo》 在软件工程中,设计模式是一种解决常见问题的标准方案,它提供了一种在特定情况下组织代码的模板。策略模式是设计模式中的一种行为模式,它允许在运行时选择算法或...

    HeadFirst 设计模式学习笔记3--装饰模式 Demo

    在“HeadFirst 设计模式学习笔记3--装饰模式 Demo”中,作者通过实例讲解了装饰模式的基本概念、结构和应用场景。这篇文章可能是从CSDN博客平台上的一个链接访问的,遗憾的是,由于我们当前无法直接访问该链接,所以...

    HeadFirst设计模式读书笔记

    HeadFirst设计模式 读书 笔记

    HeadFirst 设计模式学习笔记2--观察者模式 demo

    总的来说,HeadFirst设计模式的学习笔记2关于观察者模式的演示,旨在帮助开发者理解如何使用观察者模式来构建可扩展的系统。通过实际的代码示例,我们可以更深入地掌握这一模式,并将其应用到日常开发中,提升代码的...

    读书笔记:设计模式学习笔记和代码。《图解设计模式》《Head First 设计模式》.zip

    读书笔记:设计模式学习笔记和代码。《图解设计模式》《Head First 设计模式》

    Head First 设计模式 扫描版

    《Head First 设计模式》是软件开发领域内一本广受欢迎的书籍,由Eric Freeman、Elisabeth Robson、Bert Bates和Kathy Sierra四位作者共同撰写。这本书以其独特的视觉风格和易于理解的教学方法,深入浅出地介绍了...

    Head First设计模式读书笔记-DesignPatterns.zip

    《Head First设计模式》是一本深受开发者喜爱的设计模式学习书籍,它以易懂且生动的方式介绍了23种经典设计模式。这些模式是软件工程中经过实践验证的最佳实践,旨在提高代码的可重用性、可读性和可维护性。下面,...

    HeadFirst设计模式笔记

    《HeadFirst设计模式笔记》是深入理解软件设计思想的一份宝贵资料,主要涵盖了设计模式这一核心编程概念。设计模式是经过实践验证的解决方案模板,用于解决在软件开发中经常遇到的问题,尤其在面向对象设计中。这篇...

    Head First 设计模式学习笔记(十四)模式的组合使用

    在《Head First 设计模式学习笔记(十四)模式的组合使用》中,作者探讨了如何在实际编程中灵活地组合多种设计模式以解决复杂问题。这篇文章可能是基于《Head First 设计模式》这本书的一个章节,该书是设计模式领域...

    《Head First设计模式》读书笔记 -- (第一章)策略模式

    《Head First设计模式》是软件开发领域的一本经典著作,其深入浅出地介绍了23种设计模式。第一章主要讲解的是策略模式,这是一种行为设计模式,它使你能在运行时改变对象的行为。策略模式的核心思想是定义一系列算法...

    基于Java语言的《Head First 设计模式》学习笔记及实战练习源码

    本项目为《Head First 设计模式》的Java语言学习笔记与实战练习源码集合,包含104个文件,主要包括88个Java源文件、12个Markdown文档、3个XML配置文件及少量其他辅助文件。内容涵盖设计模式的学习笔记以及相应的代码...

    Head First Design Pattern 学习笔记

    著名的《Head First Design ...由于书本过长,整理出笔记帮助回想起设计模式。文件是docx格式,只能由OFFICE Word 2007之后的版本打开,内附Visio类图文件。本文由个人整理摘录,部分内容来自书本,仅供学习使用。

    head first 设计模式

    根据提供的信息,“Head First设计模式”是一本广受好评的设计模式书籍。虽然具体的章节内容没有给出,但从描述中得知本书的第22页至39页涵盖了重要的设计模式概念,因此我们将围绕这些页面可能涉及的设计模式进行...

    Head First Servlets & JSP 学习笔记

    以上只是《Head First Servlets & JSP》一书中的部分核心知识点,实际内容还包括过滤器、监听器、MVC设计模式、JSTL等更广泛的主题,旨在帮助读者全面理解和掌握Servlet和JSP技术。通过深入学习,开发者能够构建高效...

    head_first_servlet&jsp学习笔记

    【Servlet&JSP基础知识】 ...以上是`head_first_servlet&jsp`学习笔记的主要知识点,涵盖了Servlet和JSP的基础、Web应用架构、MVC模式、会话管理和JSP编程等多个方面,为深入理解和实践Servlet与JSP开发奠定了基础。

Global site tag (gtag.js) - Google Analytics