java单一窗体设计模式,小菜学设计模式——单例模式

背景

保证只有一个实例时,你是否会想到单例模式

1、使用意图

保证只有一个实例,比如帮助窗体,没必要维护多个实例

2、生活实例

地球只有一个(不知道算不算单例的体现)

3、Java 例子(框架、JDK 、JEE)

单例模式的例子数不胜数,很常见的例子就是Spring的Controller是一个单例,但是如果按照单例模式编码规则来说,Servlet不是单例,但是他确实单实例,意思是Web容器只维护一个Servlet对象,其实也可以等同为单例的结果。另外,也不要一看到类名.getInstance()就以为那是单例模式,比如 Calendar.getInstance() 他不是单例模式,得到的对象都是不同是的实例。

/**

* Gets a calendar using the default time zone and locale. The

Calendar returned is based on the current time

* in the default time zone with the default locale.

*

* @return a Calendar.

*/

public static Calendar getInstance()

{

Calendar cal = createCalendar(TimeZone.getDefaultRef(), Locale.getDefault());

cal.sharedZone = true;

return cal;

}

4、模式类图

850e0abe185e055f351d708e40172647.png

617dd19a7751e2fe2e626303ed7983e8.png

5、模式优点

单例模式(Singleton),保证一个类仅有一个实例,并提供一个访问它的全局访问点。

通常我们可以让一个全局变量使得一个对象被访问,但它不能防止你实例化多个对象。一个最好的办法就是,让类自身负责保存它的唯一实例。这个类可以保证没有其他实例可以被创建,并且它可以提供一个访问该实例的方法。

因为Singleton类封装它的唯一实例,这样它可以严格地控制客户怎样访问它以及何时访问它。简单的说就是对唯一实例的受控访问。

单例模式编写最关键的一点是保证方法私有,同时获取实例时保证只有一个对象实例。

6、与类似模式比较

单例模式中主要需要解决的是两个问题:全局访问和实例化控制问题。

静态初始化方式是在自己被加载时就将自己实例化,被形象的称为饿汉式单例类。这种方式是类一加载就实例化的对象,要提前占用系统资源。

原先的单例模式处理方式第要在第一次被引用时,才会将自己实例化,所以称为懒汉式单例类。该方式面临着多线程访问的安全性问题,需要双重锁定这样的处理才能达到保证安全。

谈到单例,我觉得他和静态类还是要说说的,比如数学类这个静态类,静态类一加载,所有方法都会被加载到内存中,他是一种无状态的类,也就是没有实例拥有自己独有的属性,这种类一般就是工具类;然而,单例则不同,他自己内部保存了一份实例,这个实例就是维持了自身的状态,所以这是不同的;另外,因为静态所有的方法都为静态,所以,不能继承,但是单例不一样,单例的方法同样都不是静态的,他是可以继承的。二者共同点是构造方法都私有化。

7、代码实现

1)普通单例(一般单例只要保证构造方法私有),这种单例不安全,原因是多线程访问可能导致产生不止一个实例

/**

* Singleton model (懒汉式单例)

*@author:Heweipo

*@version 1.00

*

*/

public class Singleton {

private static Object lock = new Object();

private static Singleton instance;

private Singleton(){

// do nothing

}

public static Singleton getInstance(){

if(instance == null){

instance = new Singleton();

}

return instance;

}

}

2)多线程单例,Double-Check Locking 双重锁定(在第一次被引用时,才会将自己实例化,所以也称之为 懒汉单例)

/**

* Singleton model (懒汉式单例)

*@author:Heweipo

*@version 1.00

*

*/

public class Singleton {

private static Object lock = new Object();

private static Singleton instance;

private Singleton(){

// do nothing

}

public static Singleton getInstance(){

// 其实可以在方法最外层枷锁,但是那样的话势必会影响性能,因为为空的情况只有一次出现

if(instance == null){

// 枷锁lock,在这之前是可能有多个方法还未初始化单例时进入的,所以后面还有判断一次

synchronized(lock){

// 如果引用为空,那么实例化单例,这里为什么还要判断呢?目的是防止多个方法同时进入

if(instance == null){

//单例首次实例化

instance = new Singleton();

}

}

}

return instance;

}

}

3)静态初始化(在自己被加载时就将自己实例化,被称为饿汉式单例类,提前占用系统资源,不过更加建议使用这种,简洁)

/**

* Singleton model (饿汗式单例)

*@author:Heweipo

*@version 1.00

*

*/

public class Singleton {

// 类加载时就实例化对象

private static Singleton instance = new Singleton();

private Singleton(){

// do nothing

}

public static Singleton getInstance(){

// 直接返回单例

return instance;

}

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值