设计模式(4) -- 单例模式(Singleton)
试想一个读取配置文件的需求,创建完读取类后通过New一个类的实例来读取配置文件的内容,在系统运行期间,系统中会存在很多个该类的实例对象,也就是说系统中会同时存在多份配置文件的内容,这样会严重浪费内存资源。这样需要实现:在一个系统运行期间,只要一个类实例就可以了。
单例模式的定义:保证一个类仅有一个实例,并提供一个访问它的全局访问点。
现在一个类能够被创建多个实例,问题的根源在于类的构造方法是公开的,也就是可以让类的外部通过构造方法创建多个实例。换句话说,只要类的构造方法能让类的外部访问,就没有办法去控制外部来创建这个类的实例个数。
要想控制一个类只被创建一个实例,那么首要的问题就是要把创建实例的权限收回来,让类自身来负责自己类实例的创建工作,然后由这个类来提供外部可以访问这个类实例的方法,这就是单例模式的实现方式。
在Java中,单例模式分为两种,懒汉式和饿汉式。
懒汉式:
public
classSingletonLan {
privatestaticSingletonLan
uniqueInstance = null;
privateSingletonLan(){
}
publicstaticsynchronizedSingletonLan getInstance(){
if(uniqueInstance==null){
uniqueInstance =
new SingletonLan();
}
return
uniqueInstance;
}
publicvoidsingletonOPeration(){
}
privateString
singletonData;
publicStringgetSingletonData(){
return
singletonData;
}
}
饿汉式:
public
classSingletonE {
private
static SingletonE uniqueInstance=
newSingletonE();
privateSingletonE(){
}
publicstaticSingletonEgetInstance(){
return
uniqueInstance;
}
publicvoidsingletonOPeration(){
}
privateString
singletonData;
publicString getSingletonData(){
return
singletonData;
}
}
Java里面实现的单例是一个虚拟机范围。因为装载类的功能是虚拟机的,所以一个虚拟机在通过自己的ClassLoader装载饿汉式实现单例类的时候就会创建一个类的实例。这就意味着如果一个虚拟机里面有很多个ClassLoader,而且这些ClassLoader都装载某个类的话,就算这个类是单例,它也会产生很多个实例。当然,如果一个机器上有多个虚拟机,那么每个虚拟机里面都应该至少有一个这个类的实例,也就是说整个机器上就有很多个实例,更不会是单例了。
单例模式的精粹是static变量在类装载的时候进行初始化;多个实例的static变量会共享同一块内存区域。
单例模式有一种应用叫延迟加载(Lazy Load):一开始不需要加载资源或者数据,一直等,等到马上要使用这个资源或者数据的时候才去加载。
利用缓存来实现单例模式:
import java.util.HashMap;
import java.util.Map;
public
classSingletonCache {
private
final static String DEFAULT_KEY="One";
private
static Map<Object,SingletonCache> map =
newHashMap<Object,SingletonCache>();
private SingletonCache(){
}
public
static SingletonCache getInstance(){
SingletonCache instance = (SingletonCache)map.get(DEFAULT_KEY);
if(instance==null){
instance = new SingletonCache();
map.put(DEFAULT_KEY,instance);
}
return instance;
}
}
从线程安全上讲,懒汉式是线程不安全的,饿汉式是线程安全的(虚拟机保证只会加载一次,在装载类的时候是不会发生并发的),但是加了synchronized关键字的懒汉式是线程安全的,只是效率下降了。
可以使用双重检查加锁机制,指的是并不是每次进入getInstance方法都需要同步,而是先不同步,进入方法过后,先检查实例是否存在,如果不存在才进入下面的同步块,这是第一重检查。进入同步块后,再次检查实例是否存在,如果不存在,就在同步的情况下创建一个实例,这是第二重检查。这样一来,就只需要同步一次了。
双重检查加锁机制的实现用到关键字volatile,意思是被volatile修饰的变量的值,将不会被本地线程缓存,所有对该变量的读写都是直接操作共享内存,从而确保多个线程能正确的处理该变量。由于volatile关键字可能会屏蔽虚拟机中一些必要的代码优化,所以运行效率并不是很高。
public
classSingletonVolatile {
privatevolatilestaticSingletonVolatile
instance = null;
privateSingletonVolatile(){
}
publicstaticSingletonVolatilegetInstance(){
if(instance==
null){
synchronized(SingletonVolatile.class){
if(instance==null){
instance =
newSingletonVolatile();
}
}
}
return
instance;
}
}
Lazyinitialization holder class模式综合实现了延迟加载和线程安全。在类加载的时候不去初始化对象。
public
classSingletonClass {
privatestaticclassSingletonHolder{
private
staticSingletonClass instance =
newSingletonClass();
}
privateSingletonClass(){
}
publicstaticSingletonClassgetInstance(){
return SingletonHolder.instance;
}
}
据说单元素的枚举类型已经成为实现Singleton的最佳方法。
分享到:
相关推荐
NULL 博文链接:https://linxingliang.iteye.com/blog/1217811
设计模式总结-模板设计模式,单例模式(singleTon)
设计模式系列之01-单例模式(Singleton模式),很好的资源,理论实践结合讲述,逐步更新
设计模式C++学习之单例模式(Singleton)
Java面向对象(高级)-- 单例(Singleton)设计模式
单例模式(Singleton Pattern)是 Java 中最简单的设计模式之一。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。 这种模式涉及到一个单一的类,该类负责创建自己的对象,同时确保只有单个...
单例设计模式Singleton1
简单的单例模式举例Singleton 分为恶汉式 懒汉式
这是单例设计模式的基本示例。 您将在此仓库中找到两个示例。 首先是线程安全的单例,其次不是。 测试的输出是: synchronizedTest Started... Thread 2 instance is : ...
本文档,是利用C++来实现设计模式中的,单例模式,里面有内容说明和相关实例代码介绍
在个人自学阶段的23种设计模式代码的全部实现,全部使用Java编写,其中还包括各个设计模式在源码中的使用,每种设计模式都举了一个简单的小例子来进行实现,并加以注释 包名解释 一、DesignPattern 1.1 创建型模式 ...
C# 23种设计模式之单例模式源码
java设计模式,单例模式的不同实现方式
- 单例模式(Singleton) - 建造者模式(Builder) - 原型模式(Prototype) - 代理模式(Proxy) - 适配器模式(Adapter) - 装饰器模式(Decorator) - 桥接模式(Bridge) - 组合模式(Composite) - 外观模式(Facade) ...
IOS 单例设计模式实例Demo 单例 设计 模式 IOS Singleton
单例模式(Singleton Pattern)是 Java 中最常见的设计模式之一。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。
delphi 单例模式 设计模式中的单例模式 与其他语言相似
设计模式里面的单例模式程序 package com.rrppff; public class Singleton { private static String name; public static String getName() { return name; } public static void setName(String name) { ...
php /** * 单例模式 * * 保证一个类仅有一个实例,并提供一个访问它的全局访问点 * */ class Singleton { static private $_instance = null; private function __construct() { } static public function ...
一个产生随机数的例子,整个应用程序中只需要一个类的实例来产生随机数,客户端程序从类中获取这个实例,调用这个实例的方法nextInt(),公用的方法访问需要进行同步,这是单例模式需要解决的同步问题。