饿汉式单例
package com.tencent.baosen; /** * 面试题:请编写一个Singleton程序,并说明其主要特点? * 1.代码如上,可以把懒汉式和饿汉式都写上。 * 2. 懒汉式单例高级实现,要考虑到线程同步的问题,请参考Singleton3的实现 * 3. 主要特点是构造方法私有化,类内部提供static方法获取实例化对象,这边不管外部如何操作,永远都只有一个实例化对象 * * 单例设计模式开发学习 * 单例的一个特征是构造方法私有化 * 饿汉式单例实现 * 饿汉式的特征是直接在声明单例属性时就实例化,无需调用getInstance方法时,单例实例化 * 饿汉式:在系统加载类的时候就会自动提供有Singleton类的实例化对象。 * 懒汉式:在第一次使用的时候才进行实例化对象。 */ public class Singleton2 { private static final Singleton2 INSTANCE = new Singleton2(); private Singleton2(){ } public static Singleton2 getInstance(){ return INSTANCE; } public String print() { return "我是饿汉式单例实现"; } }简单的懒汉式单例
package com.tencent.baosen; /** * 单例设计模式开发学习 * 由于某些实现要求,只允许某个类只提供一个实例化对象(例如某些工具类),也就是说在很多情况下,有些类是不需要重复产生对象的 * 所以构造方法必须私有化,外部的类不能调用构造进行实例化,才能保证实例化对象不是多个。 * 单例的一个特征是构造方法私有化,构造方法私有化后,外部的类无法调用构造方法进行实例化 * 懒汉式单例实现, * 懒汉式的特征是在调用getInstance方法时,单例才实例化 */ public class Singleton1 { private static Singleton1 instance = null; private Singleton1(){} //构造方法私有化了, public static Singleton1 getInstance(){ if (instance == null){ instance = new Singleton1(); } return instance; } public String print() { return "我是懒汉式单例实现"; } }线程同步的懒汉式单例实现
package com.tencent.baosen; /** * 单例学习 * 线程安全的单例实现,是在懒汉式单例实现的基础上实现的。 * 面试题:请编写单例设计模式 * 1. [100%] 直接编写一个饿汉式的单例设计模式,并且实现构造方法私有化 * 2. [120%] 在Java中哪里使用到了单例设计模式? Runtime类,Spring框架; * 3. [200%] 懒汉式单例设计模式的问题? 线程不同步。 */ public class Singleton3 { //为什么加volatile? instance=new Singleton3()被实例化时应该立刻与主内存中的数据对象保持同步,而不应该存副本,所以加volatile private static volatile Singleton3 instance = null; private Singleton3(){ } // 如果将synchronized关键字加在getInstance方法上,同步效率会变低。 public static Singleton3 getInstance(){ if(instance == null){ //多个线程可能同时通过这个判断 synchronized (Singleton3.class){ if(instance == null){ instance = new Singleton3(); } } } return instance; } public String print(){ return "我是线程同步的单例实现类"; } }测试类:
package com.tencent.baosen; public class Test { public static void main(String[] args) { Singleton1 single1 = Singleton1.getInstance(); System.out.println("single1 = "+ single1.print()); Singleton2 single2 = Singleton2.getInstance(); System.out.println("single2 = "+ single2.print()); Singleton3 single3 = Singleton3.getInstance(); System.out.println("single3 = "+ single3.print()); } }