六种常用的设计模式java实现(二)单例模式
来源:互联网 发布:郑州淘宝店运营公司 编辑:程序博客网 时间:2024/06/01 15:15
上一节说完了工厂模式,那么,接下来给大家介绍下单例模式。单例模式想必大家应该都用得比较多,特别是在spring中,我们的java bean对象一般都是单例的。像项目中某些类只会被实例化一次或者只能实例化一次,我们都可以把这个类做成单例的。
一、基本概念
单例模式又叫做单态模式或者单件模式,保证一个类仅有 一个实例,并提供一个访问它的全局访问点。单例模式中的“单例”通常用来代表那些本质上 具有唯一性的系统组件(或者叫做资源)。比如文件系统、资源管理器等等。单例模式的目的就是要控制特定的类只产生一个对象,当然也允许在一定情况下灵活的 改变对象的个数。
二、分类及含义
单例模式的分类有好几种,把以下三种搞清楚就够了。
饿汉式:在第一次调用的时候实例化自己
懒汉式:在类初始化时,已经自行实例化,三种实现,synchronized、双重检查锁定、静态内部类
登记式:类似Spring里面的方法,将类名注册,下次从里面直接获取
三、代码实现
1、饿汉式
这里,我先直接贴出代码:
package org.wuqiong.designpattern.singlepattern.hungryType;/** * ClassName:Singleton <br/> * Function: 单例模式-饿汉式. <br/> * Date: 2016年1月28日 上午11:57:28 <br/> * @author qiyongkang * @version * @since JDK 1.6 * @see */public class Singleton { private static final Singleton instance = new Singleton(); private Singleton() {} public static Singleton getInstance() { return instance; }}
可以看出,在此类被加载的时候,只被实例化一次。所以,这种方式,无论单线程还是多线程,都是比较安全的。唯一的缺点就是类一被加载就会实例化一次。
2、懒汉式
这里,懒汉式实现有三种方法:
1、使用synchronized同步方法
/** * ClassName:Singleton <br/> * Function: 单例模式-懒汉式,synchronized. <br/> * Date: 2016年1月28日 下午12:10:15 <br/> * @author qiyongkang * @version * @since JDK 1.6 * @see */public class Singleton { private static Singleton instance = null; private Singleton() {} /** * * getInstance:在getInstance方法上加同步. <br/> * * @author qiyongkang * @return * @since JDK 1.6 */ public synchronized static Singleton getInstance() { if (instance == null) { instance = new Singleton(); } return instance; }}
这里,在方法前加synchronized关键字的目的大家应该清楚,就是防止多线程第一次实例化的时候出现问题。很显然,这种方式就没有在类加载的时候便实例化一次,而是在要用的时候才开始实例化。但是,由于加了同步,对性能方面也会有所影响。
2、双重检查锁定
package org.wuqiong.designpattern.singlepattern.lazyType;/** * ClassName:单例模式-懒汉式,双重检查锁定<br/> * Function: TODO ADD FUNCTION. <br/> * Date: 2016年1月29日 上午11:59:15 <br/> * @author qiyongkang * @version * @since JDK 1.6 * @see */public class Singleton2 { /** * volatile 变量.用来确保将变量的更新操作通知到其他线程,保证了新值能立即同步到主内存,以及每次使用前立即从主内存刷新 */ private static volatile Singleton2 instance = null; private Singleton2() {} /** * * getInstance: 双重检查锁定. <br/> * * @author qiyongkang * @return * @since JDK 1.6 */ public static Singleton2 getInstance() { if (instance == null) { synchronized (Singleton2.class) { if (instance == null) { instance = new Singleton2(); } } } return instance; }}
这里,对比上一种方法,在性能方面有所改进,将同步关键字放在方法里面。然后,就是为什么使用volatile关键字以及两次为空判断,大家可以仔细体会一下,这个主要是防止在多线程中出现问题。
3、静态内部类
package org.wuqiong.designpattern.singlepattern.lazyType;/** * ClassName:Singleton3 <br/> * Function: 单例模式-懒汉式,静态内部类. <br/> * Date: 2016年1月29日 下午2:09:31 <br/> * @author qiyongkang * @version * @since JDK 1.6 * @see */public class Singleton3 { private static class LazyHolder { private static final Singleton3 INSTANCE = new Singleton3(); } private Singleton3() {} /** * * getInstance:静态内部类. <br/> * * @author qiyongkang * @return * @since JDK 1.6 */ public static final Singleton3 getInstance() { return LazyHolder.INSTANCE; }}
这种方法,大家可以好好体会一下,应该比以上两种方法都要好,一般比较推荐使用这种方法。
3、登记式
/** * ClassName:Singleton <br/> * Function: 单例模式-登记式 <br/> * Reason: TODO ADD REASON. <br/> * Date: 2016年1月29日 下午2:16:36 <br/> * * @author qiyongkang * @version * @since JDK 1.6 * @see */public class Singleton { private static Map<String, Singleton> map = new HashMap<String, Singleton>(); static { Singleton single = new Singleton(); map.put(single.getClass().getName(), single); } // 保护的默认构造子 protected Singleton() { } // 静态工厂方法,返还此类惟一的实例 public static Singleton getInstance(String name) { if (name == null) { name = Singleton.class.getName(); System.out.println("name == null" + "--->name=" + name); } if (map.get(name) == null) { try { map.put(name, (Singleton) Class.forName(name).newInstance()); } catch (InstantiationException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (ClassNotFoundException e) { e.printStackTrace(); } } return map.get(name); } // 一个示意性的商业方法 public String about() { return "Hello, I am RegSingleton."; } public static void main(String[] args) { Singleton single3 = Singleton.getInstance(null); System.out.println(single3.about()); }}
仔细看看,其实登记式和饿汗式差不多,也是在类加载的时候实例化一次,只不过会把此对象放入map中注册一下。这种方式,其实和spring中的做法很相似,都是通过反射的机制实例化对象。
那么,单例模式就讲到这儿了,希望对单例模式还不太了解的童鞋提供点帮助!
- 六种常用的设计模式java实现(二)单例模式
- 六种常用的设计模式java实现(六)策略模式
- 六种常用的设计模式java实现(一)工厂模式
- 六种常用的设计模式java实现(三)责任链模式
- 六种常用的设计模式java实现(四)模板模式
- 六种常用的设计模式java实现(五)代理模式
- AspectJ实现设计模式(六)—单例模式
- AspectJ实现设计模式(六)—单例模式
- Java设计模式(二):单例模式的5种实现方式,以及在多线程环境下5种创建单例模式的效率
- Java设计模式(二):单例模式的5种实现方式,以及在多线程环境下5种创建单例模式的效率
- Java单例设计模式(二)
- 常用设计模式-单例的几种实现方式
- java 常用设计模式------单例模式
- java中常用的设计模式-单例模式
- Java中常用的设计模式之单例模式
- java单例设计模式的实现
- Java设计模式之二——四种不同的单例模式(Singleton)
- 【设计模式】常用Java设计模式之单例模式
- [LeetCode] Increasing Triplet Subsequence
- Java基础03 构造器与方法重载
- 岁月划过生命线——大二下
- 常量和变量
- 杂谈之C++中的“错误”处理
- 六种常用的设计模式java实现(二)单例模式
- canvas+video实现跨域视频截图
- 20160122.CCPP详解体系(0001天)
- 14.RunLoop的理解
- 大型互联网架构案例-LinkedIn系统架构
- C++虚函数
- XUtils3框架最最最简单的使用
- Memory Leak检测神器--LeakCanary初探
- 编程题#2:输出指定结果一(C++程序设计第10周)