JAVA设计模式--辛格尔顿

Singleton模式可以作为一种编程技术,让我们先从理论上说代码

单例模式三个关键点:

1)、某个类仅仅能有一个实例

2)、该类必须自行创建这个实例

3)、该类必须自行向整个系统提供这个实例

应用场景:

1)、window的任务管理器就是非常典型的单例模式,你肯定不能同一时候打开两个任务管理器

2)、数据库连接池技术一般採用的都是单例模式。由于数据库连接是一种数据库资源。系统中使用数据库连接池,主要是节省打开或者关闭数据库连接所引起的

效率损耗。这样的效率上的损耗还是很昂贵的,用单例模式来维护。就能够大大减少这样的损耗。

3)、我们在进行开发时对于配置文件的读取一般也是採用单例模式,由于配置文件里的内容是全局共享的资源。

4)、多线程的线程池设计一般也要考虑单例模式。线程池能方便对池中线程的控制。

5)、站点的统计类信息。一般也是採用单例模式,否则难以同步控制。比如统计我的博客的訪问量。

6)、我们开发应用程序的日志功能也是採用的单例模式,由于我们仅仅能有一个实例去追加日志信息,否则不好控制。

单例模式的几类写法:

1)饿汉式模式

怎么理解呢,饿了吗。所以我们做饭要很着急,这里也就是说。当类被载入的时候该类就已经将自己的实例创建出来了。

这也是空间换取时间的典型应用,怎么说呢? 我们在类载入的时候就实例化了这个对象。占用了内存空间,可是我们在用到这个对象的时候就不用去实例化了,

直接拿去用就能够了。也就节省可时间,这也就是空间换取时间。

ps:记得第一份工作(还在大三的时候)面试的时候面试官就让我举出我做过的项目中时间换取空间和空间换取时间的典型应用,给我问懵了。

代码:

package chc.singleton;
/**
 * 饿汉式单例模式类
 * @author haicheng.cao
 * @time 2014.09.02 22:40
 */
public class EagerSingleton {
	//类载入的时候就创建了自身的实例对象--饿汉式(空间换取时间)
	public static EagerSingleton eagerSingleton=new EagerSingleton();

	/**
	 * 显示的私有构造方法,防止其它类创建本类实例
	 */
	private EagerSingleton(){

	}

	/**
	 * 静态工厂方法,其它类通过调用该方法来获取本类的实例对象
	 */
	public static EagerSingleton getEagerSingleton(){
		return eagerSingleton;
	}
}

2)懒汉式模式

这个怎么理解呢?懒人吗,举个样例。一个人立即要去面试的时候才開始写简历,就是说对象实例要用的时候才去创建。在这里也就是说在类载入的时候并没有创

建本类的实例对象,而是在其它类在第一次调用的时候才去创建。

这也是典型的时间换取空间的应用,就是嘛,类载入的时候没有创建对象。节省了内存。也就是节省了空间,调用的时候须要推断一下这个类的实例对象是否存

在,浪费了时间,这也就是时间换取空间。

代码:

package chc.singleton;
/**
 * 懒汉式单例模式类
 * @author haicheng.cao
 * @time 2014.09.02 23:05
 */
public class LazySingleton {
	//类载入的时候并没有创建自身实例化对象
	public static LazySingleton lazySingleton=null;

	/**
	 * 显示的私有构造方法。防止其它类创建本类实例
	 */
	private LazySingleton(){

	}

	/**
	 * 静态工厂方法,其它类通过调用该方法来获取本类的实例对象
	 */
	public static synchronized LazySingleton getLazySingleton(){
		//第一次被调用的时候创建自身实例对象
		if(lazySingleton==null){
			lazySingleton=new LazySingleton();
		}
		return lazySingleton;
	}
}

注意:我们给getEasySingleton()方法加了同步keyword,能够保证线程安全。可是这样做比較影响程序的性能,我们能不能改善一下呢?看以下的:

3)双重检查加锁

双重检查加锁机制的意思就是:我们在调用getEasySingleton()方法的时候不同步,进入方法内我们推断一下实例对象是否存在。假设不存在我们在进入同步代

码块。这是第一重检查,进入同步块后再进行推断,推断实例是否存在,假设不存在再创建这个对象的实例。这就是第二重检查。

这样。就仅仅有第一次调用的时候

运行了一次同步代码块,其余的时候就不须要同步了,提升了程序的性能。

代码:

package chc.singleton;
/**
 * 双重检查加锁。针对懒汉式提升性能
 * @author haicheng.cao
 * @time 2014.09.02 22:40
 */
public class TwoCheck {
    private volatile static TwoCheck twoCheck = null;

    private TwoCheck(){

    }

    public static TwoCheck getInstance(){
        //先检查实例是否存在,假设不存在才进入以下的同步块
        if(twoCheck == null){
            //同步块,线程安全的创建实例
            synchronized (TwoCheck.class) {
                //再次检查实例是否存在,假设不存在才真正的创建实例
                if(twoCheck == null){
                	twoCheck = new TwoCheck();
                }
            }
        }
        return twoCheck;
    }
}

-------------------------------------以下续写与2014.09.03 21:15-----------------------------------------------------

昨天在写这个东西的时候一直在纠结一个问题,如果有一部分全局共享的变量,我们能够通过在类中声明静态属性。然后通过静态方法来初始化声明的那些属性。

然后这些静态的变量在不论什么一个类中都能够被调用了。就像以下这种代码:

package chc.statics;
public class StaticDemo {
	public String logPath=null;

	public void init(){
		logPath="c://log.txt";
	}
}

这种代码不是全然能够替代单例的功能吗?

今天上班问了下领导。给我解释的非常清楚,类中声明静态变量的方式的确能够实现单例模式的功能。可是,上面代码那种方式你须要在项目启动的时候调用一下

StaticDemo类的init()方法。单例模式就是全然由自身去维护自己,不须要借助外力。

另一种情况:就是当有些全局属性是动态变化的时候,那么对于静态变量的方式就须要程序不断的去操作该类的动态属性,而静态类能够灵活的自己控制。解除

了代码的耦合。

package chc.singleton;
import java.io.*;
public class Singleton {
	public static File file=null;

	public static long lastModified;
	private static Singleton s=null;

	private Singleton(){
		lastModified=file.lastModified();
	}
	public synchronized static Singleton getSingleton() {
		//假设变量lastModified的值与文件最后一次被改动的时间值不同的话,又一次实例化一下
		if(s==null && lastModified!=file.lastModified() ){
			s=new Singleton();
		}
		return s;
	}
}

这个样例就非常直观了。类中的lastModified的值是动态的,假设用静态代码块去维护的话。程序在每一次改动这个文件的时候都要调用一次静态代码

块又一次初始化一下这个变量,单例中却能够自己灵活的进行维护,不须要别的类辅助。

-------------------------------------以下续写与2014.11.03 20:40-----------------------------------------------------

饿汉式存在着占用资源的问题。懒汉式存在着线程安全的问题,以下看一个巧妙的写法,将懒汉式与饿汉式的长处集成在了一起。攻克了懒汉式与饿汉式的弊端。

package hirain;

import java.io.BufferedInputStream;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.Properties;
/**
 * 即实现了延迟载入,又线程安全
 * @author haicheng.cao
 */
public class AppConfig5 {

	//静态内部类在第一次使用的时候被装载
	 private static class AppConfig5Holder{
	        private static AppConfig5 instance = new AppConfig5();
	 }
	/**
	 * 定义一个方法来为client提供AppConfig类的实例
	 * @return 一个AppConfig的实例
	 */
	public static  AppConfig5 getInstance(){
		return AppConfig5Holder.instance;
	}
	/**
	 * 私有化构造方法
	 */
	private AppConfig5(){
		//调用读取配置文件的方法
		readConfig();
	}

	private String id;

	private String name;

	public String getId() {
		return id;
	}
	public void setId(String id) {
		this.id = id;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}	

	/**
	 * 读取配置文件,把配置文件里的内容读出来设置到属性上
	 */
	private void readConfig(){
		Properties p = new Properties();
		InputStream in = null;
		try {
			in = new BufferedInputStream (new FileInputStream("AppConfig.properties"));
			p.load(in);
			//把配置文件里的内容读出来设置到属性上
			this.id = p.getProperty("id");
			this.name = p.getProperty("name");
		} catch (IOException e) {
			System.out.println("装载配置文件出错了,详细堆栈信息例如以下:");
			e.printStackTrace();
		}finally{
			try {
				in.close();
			} catch (IOException e) {
				e.printStackTrace();
			}
		}
	}

}

关键点:静态内部类的使用,静态内部类的静态变量仅仅有在静态内部类被使用的时候才会载入中。

版权声明:本文博主原创文章,博客,未经同意不得转载。

时间: 2024-11-05 15:47:35

JAVA设计模式--辛格尔顿的相关文章

Java设计模式(十) 备忘录模式 状态模式

(十九)备忘录模式 备忘录模式目的是保存一个对象的某个状态,在适当的时候恢复这个对象. class Memento{ private String value; public Memento(String value){ this.value = value; } public String getValue() { return value; } public void setValue(String value) { this.value = value; } } class Storage

Java设计模式(二)-单例模式

单例模式建议保证某个对象仅仅只有一个实例,当只有一个对象来协调整个系统的操作时,在计算机系统中,线程池.缓存.日志对象.对话框.打印机.显卡的驱动程序对象常被设计成单例,总之,选择单例模式就是为了避免不一致状态,避免政出多头. 下面是单例模式的类图:包括的private和static修饰的实例instance.一个private的构造函数.一个static的getInstance函数 单例模式主要有三种:懒汉式单例.饿汉式单例.登记式单例三种 1.饿汉式单例:在类的初始化时候,自行创建了实例 c

JAVA设计模式之 装饰模式【Decorator Pattern】

一.概述 动态地给一个对象增加一些额外的职责,就增加对象功能来说,装饰模式比生成子类实现更为灵活.装饰模式是一种对象结构型模式.装饰模式是一种用于替代继承的技术,使用对象之间的关联关系取代类之间的继承关系.在装饰模式中引入了装饰类,在装饰类中既可以调用待装饰的原有类的方法,还可以增加新的方法,以扩充原有类的功能. 二.适用场景 装饰原有对象.在不改变原有对象的情况下扩展增强新功能/新特征..当不能采用继承的方式对系统进行扩展或者采用继承不利于系统扩展和维护时可以使用装饰模式. 三.UML类图 四

(九)JAVA设计模式之单例模式

JAVA设计模式之单例模式 一.单例模式的介绍 Singleton是一种创建型模式,指某个类采用Singleton模式,则在这个类被创建后,只可能产生一个实例供外部访问,并且提供一个全局的访问点.     全局对象和Singleton模式有本质的区别,因为大量使用全局对象会使得程序质量降低,而且有些编程语言根本不支持全局变量.最重要的是传统的全局对象并不能阻止一个类被实例化多次. 二.单例模式的特点 单例类只能有一个实例 单例类必须自己创建自己的唯一实例. 单例类必须给所有其他对象提供这一实例.

JAVA设计模式有感

记得在大学学JAVA那阵,就觉得JAVA是一个c语言的一个包装,各种封装好的类库.功能,学习JAVA能快速上手编码.时隔8年,我感觉自己不懂JAVA. 最近看了有1个月的设计模式了,进度比较缓慢,一共才看了4个"策略模式","观察者模式","装饰者模式","工厂模式".也谈一谈自己的感受.其实这些设计模式以前也看过,当时的水准也就是硬生生记下了每个设计模式是怎么运作的,从没有想过它遵循了什么原则,达到了什么目的.因此,看过一遍

学习Java设计模式的10条建议

设计模式在整个Java的学习路线图中扮演着承上启下的作用. 在整个软件生命周期中,唯一不变的就是变化.设计模式就是要在软件设计.编码中对现有问题的一种总结,并从中寻求应对变化的策略. 自己初次接触设计模式有以下几个感觉: 内容很抽象. 示例都能看得懂,但不知道实际中如何应用. 不理解为什么要把“好好的程序”设计成这么复杂? 转眼之间到了需要自己参与需求分析.设计,并且维护之前留下的遗产代码(Legacy Code)的时候了. 再次开始学习设计模式,有了新的收获: 站在变化的角度去看,设计模式虽然

Java设计模式(五)外观模式 桥梁模式

(九)外观模式 外观模式为子系统提供一个接口,便于使用.解决了类与类之间关系的,外观模式将类之间的关系放在一个 Facade 类中,降低了类类之间的耦合度,该模式不涉及接口. class CPU { public void startup(){ System.out.println("cpu start"); } public void shutdown(){ System.out.println("cpu stop"); } } class Memory { pu

Java设计模式(六)合成模式 享元模式

(十一)合成模式 Composite 合成模式是一组对象的组合,这些对象可以是容器对象,也可以是单对象.组对象允许包含单对象,也可以包含其他组对象,要为组合对象和单对象定义共同的行为.合成模式的意义是 保证客户端调用单对象与组合对象的一致性. class TreeNode{ private String name; private TreeNode parent; private Vector<TreeNode> children = new Vector<TreeNode>();

Java设计模式之认识阶段

设计模式是什么? 设计模式(Design pattern)是一套被反复使用.多数人知晓的.经过分类编目的.代码设计经验的总结. 其本质就是继承与接口的组合应用. 为什么要用设计模? 使用设计模式是为了可重用代码.让代码更容易被他人理解.保证代码可靠性.提高工作效率. 设计模式的分类: 设计模式按目的可分为三种类型,共23种. 创建型模式:单例模式.抽象工厂模式.建造者模式.工厂模式.原型模式. 结构型模式:适配器模式.桥接模式.装饰模式.组合模式.外观模式.享元模式.代理模式. 行为型模式:模版