单例模式
八种设计模式说明
饿汉式(线程安全)
代码
public class Hungry {
private static Hungry hungry = new Hungry();
public Hungry(){}
public static Hungry getInstance(){
return hungry;
}
}
设计思想
在类加载的时候就把对象创建并放到内存中
问题
饿汉式单例模式是线程安全的,但是该实例在类装载的时候就加入到内存中,可能会造成资源浪费.测试结果
测试结果(创建1000个)
com.yczuoxin.pattern.singleton.hungry.Hungry@46bf5757
com.yczuoxin.pattern.singleton.hungry.Hungry@46bf5757
com.yczuoxin.pattern.singleton.hungry.Hungry@46bf5757
com.yczuoxin.pattern.singleton.hungry.Hungry@46bf5757
com.yczuoxin.pattern.singleton.hungry.Hungry@46bf5757
com.yczuoxin.pattern.singleton.hungry.Hungry@46bf5757
com.yczuoxin.pattern.singleton.hungry.Hungry@46bf5757
com.yczuoxin.pattern.singleton.hungry.Hungry@46bf5757
com.yczuoxin.pattern.singleton.hungry.Hungry@46bf5757
com.yczuoxin.pattern.singleton.hungry.Hungry@46bf5757
com.yczuoxin.pattern.singleton.hungry.Hungry@46bf5757
com.yczuoxin.pattern.singleton.hungry.Hungry@46bf5757
...
cost:139ms
懒汉式(线程不安全)
代码
public class ThreadUnSafeLazy {
private static ThreadUnSafeLazy lazy;
private ThreadUnSafeLazy(){}
public static ThreadUnSafeLazy getInstance(){
if(null == lazy){
lazy = new ThreadUnSafeLazy();
}
return lazy;
}
}
设计思想
在获取对象时先去判断是否实例化过,如果没有实例化就实例化一个对象
问题
在高并发环境下,如果一个线程访问时该对象还在实例化过程中,那么就会重新再实例化一个对象,导致线程不安全问题.
测试结果(创建1000个)
com.yczuoxin.pattern.singleton.lazy.ThreadUnSafeLazy@7fc8692f
com.yczuoxin.pattern.singleton.lazy.ThreadUnSafeLazy@304a7580
com.yczuoxin.pattern.singleton.lazy.ThreadUnSafeLazy@304a7580
com.yczuoxin.pattern.singleton.lazy.ThreadUnSafeLazy@304a7580
com.yczuoxin.pattern.singleton.lazy.ThreadUnSafeLazy@304a7580
com.yczuoxin.pattern.singleton.lazy.ThreadUnSafeLazy@304a7580
com.yczuoxin.pattern.singleton.lazy.ThreadUnSafeLazy@304a7580
com.yczuoxin.pattern.singleton.lazy.ThreadUnSafeLazy@304a7580
com.yczuoxin.pattern.singleton.lazy.ThreadUnSafeLazy@304a7580
com.yczuoxin.pattern.singleton.lazy.ThreadUnSafeLazy@304a7580
...
cost:98ms
懒汉式(线程安全)
public class ThreadSafeLazy {
private static ThreadSafeLazy lazy;
private ThreadSafeLazy(){}
public static synchronized ThreadSafeLazy getInstance(){
if(null == lazy){
lazy = new ThreadSafeLazy();
}
return lazy;
}
}
设计思想
由线程不安全的懒汉式可以得出该问题出现在同时两个线程调用了getInstance()方法导致,所以给与该方法加上synchronized加锁,使得该方法只有一个线程访问,保证了线程安全.
问题
synchronized锁会降低性能.增加获取实例的时间.
测试结果(创建1000个)
com.yczuoxin.pattern.singleton.lazy.ThreadSafeLazy@72f7f615
com.yczuoxin.pattern.singleton.lazy.ThreadSafeLazy@72f7f615
com.yczuoxin.pattern.singleton.lazy.ThreadSafeLazy@72f7f615
com.yczuoxin.pattern.singleton.lazy.ThreadSafeLazy@72f7f615
com.yczuoxin.pattern.singleton.lazy.ThreadSafeLazy@72f7f615
com.yczuoxin.pattern.singleton.lazy.ThreadSafeLazy@72f7f615
com.yczuoxin.pattern.singleton.lazy.ThreadSafeLazy@72f7f615
com.yczuoxin.pattern.singleton.lazy.ThreadSafeLazy@72f7f615
com.yczuoxin.pattern.singleton.lazy.ThreadSafeLazy@72f7f615
com.yczuoxin.pattern.singleton.lazy.ThreadSafeLazy@72f7f615
com.yczuoxin.pattern.singleton.lazy.ThreadSafeLazy@72f7f615
com.yczuoxin.pattern.singleton.lazy.ThreadSafeLazy@72f7f615
...
cost:129ms
枚举类(无需测试)
代码
public enum ColorEnum {
RED,
YELLOW,
BLACK,
BLUE
}
说明
枚举类可以作为单例模式是由于其特殊的性质,他在反编译的时候已经变成了final类,并且字段都被static final修饰.所以是枚举类初始化的时候,就已经初识化值了,所以也满足了单例模式的条件.
静态内部类
代码
public class StaticInnerClass {
private StaticInnerClass(){}
static class StaticInnerClassHolder{
private static final StaticInnerClass statics = new StaticInnerClass();
}
public static StaticInnerClass getInstance(){
return StaticInnerClassHolder.statics;
}
}
设计思想
只有在使用静态内部类的时候静态内部类才会实例化,只有使用到了静态内部类才会实例化该对象.不会造成对资源的浪费.
问题
创建对象所用时间较长.
测试结果(创建1000个)
com.yczuoxin.pattern.singleton.statics.StaticInnerClass@2d5a99cd
com.yczuoxin.pattern.singleton.statics.StaticInnerClass@2d5a99cd
com.yczuoxin.pattern.singleton.statics.StaticInnerClass@2d5a99cd
com.yczuoxin.pattern.singleton.statics.StaticInnerClass@2d5a99cd
com.yczuoxin.pattern.singleton.statics.StaticInnerClass@2d5a99cd
com.yczuoxin.pattern.singleton.statics.StaticInnerClass@2d5a99cd
com.yczuoxin.pattern.singleton.statics.StaticInnerClass@2d5a99cd
com.yczuoxin.pattern.singleton.statics.StaticInnerClass@2d5a99cd
com.yczuoxin.pattern.singleton.statics.StaticInnerClass@2d5a99cd
com.yczuoxin.pattern.singleton.statics.StaticInnerClass@2d5a99cd
com.yczuoxin.pattern.singleton.statics.StaticInnerClass@2d5a99cd
...
cost:146ms
注册登记式
代码
public class Register {
private static Map registerMap = new ConcurrentHashMap<>();
private Register() {
}
static {
Register register = new Register();
registerMap.put(Register.class.getName(), register);
}
public static Object getInstance(String className) {
if (null == className){
className = "com.yczuoxin.pattern.singleton.register.Register";
}
if (!registerMap.containsKey(className)){
try {
registerMap.put(className,Class.forName(className).newInstance());
} catch (Exception e) {
System.out.println("请填写正确的类的全路径");
e.printStackTrace();
}
}
return registerMap.get(className);
}
}
设计思想
用一个容器去装载所有的对象,并在容器中用其类的限定名登记所有的对象,如果实例对象在不存在,我们注册到单例注册表中,第二次取的时候根据类的限定名去取出对应的对象.不需要重新去初始化.
问题
暂无
扩展
Spring就是利用这种方式存放各种Bean.
org.springframework.beans.factory.support.DefaultSingletonBeanRegistry#getSingleton(String)
测试结果(创建1000个)
com.yczuoxin.pattern.singleton.register.Register@4cc45e92
com.yczuoxin.pattern.singleton.register.Register@4cc45e92
com.yczuoxin.pattern.singleton.register.Register@4cc45e92
com.yczuoxin.pattern.singleton.register.Register@4cc45e92
com.yczuoxin.pattern.singleton.register.Register@4cc45e92
com.yczuoxin.pattern.singleton.register.Register@4cc45e92
com.yczuoxin.pattern.singleton.register.Register@4cc45e92
com.yczuoxin.pattern.singleton.register.Register@4cc45e92
com.yczuoxin.pattern.singleton.register.Register@4cc45e92
com.yczuoxin.pattern.singleton.register.Register@4cc45e92
...
cost:108ms
双重校验
代码
public class DoubleCheck {
private static volatile DoubleCheck doubleCheck;
private DoubleCheck(){}
public static DoubleCheck getInstance(){
if (null == doubleCheck){
synchronized (DoubleCheck.class){
if(null == doubleCheck){
doubleCheck = new DoubleCheck();
}
}
}
return doubleCheck;
}
}
技术思路
利用volite可见性和synchronized锁保证单例的创建是线程安全的.
问题
volite会使缓存失效,消耗性能,synchronized锁也导致性能的消耗,所以总的说来很耗性能.
测试结果(创建1000个)
com.yczuoxin.pattern.singleton.doublecheck.DoubleCheck@7097d410
com.yczuoxin.pattern.singleton.doublecheck.DoubleCheck@7097d410
com.yczuoxin.pattern.singleton.doublecheck.DoubleCheck@7097d410
com.yczuoxin.pattern.singleton.doublecheck.DoubleCheck@7097d410
com.yczuoxin.pattern.singleton.doublecheck.DoubleCheck@7097d410
com.yczuoxin.pattern.singleton.doublecheck.DoubleCheck@7097d410
com.yczuoxin.pattern.singleton.doublecheck.DoubleCheck@7097d410
com.yczuoxin.pattern.singleton.doublecheck.DoubleCheck@7097d410
...
cost:124ms
序列化
代码
public class Serialize implements Serializable {
private static Serialize serialize = new Serialize();
private Serialize(){}
public static Serialize getInstance(){
return serialize;
}
protected Object readResolve(){
return serialize;
}
}
测试代码
public class SerializableTest {
public static void main(String[] args) {
Serialize serialize = Serialize.getInstance();
System.out.println(serialize);
writeFile("D://serialize.txt", serialize);
Serialize serialized = (Serialize)readFile("D://serialize.txt");
System.out.println(serialized);
}
private static void writeFile(String path, Object object) {
FileOutputStream fos = null;
ObjectOutputStream oos = null;
try{
fos = new FileOutputStream(new File(path));
oos = new ObjectOutputStream(fos);
oos.writeObject(object);
} catch (Exception e){
e.printStackTrace();
} finally {
try {
oos.close();
} catch (IOException e) {
e.printStackTrace();
}
try {
fos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
private static Object readFile(String path) {
FileInputStream fis = null;
ObjectInputStream ois = null;
Object object = null;
try{
fis = new FileInputStream(new File(path));
ois = new ObjectInputStream(fis);
object = ois.readObject();
} catch (Exception e){
e.printStackTrace();
} finally {
if (null != ois){
try {
ois.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (null != fis){
try {
fis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
return object;
}
}
设计思想
利用序列化和反序列来创建对象,为了使创建的对象是单例,必须实现Serializable接口及重写readResolve(),当实现了readResolve方法后,jvm就会有readResolve返回指定对象,也就保证了单例性.
protected Object readResolve(){
return serialize;
}
缺点
使用起来比较复杂,还要使用到IO读写
测试结果
com.yczuoxin.pattern.singleton.serialize.Serialize@4554617c
com.yczuoxin.pattern.singleton.serialize.Serialize@4554617c
优点
节省资源空间,减少了new对象所消耗的性能,并且可以在任何地方拿到同样的东西.
缺点
当获取对象时不能用new关键字来创建,而要知道其API,增加了开发人员的使用难度.
使用场景
工具类
日志等文件系统
各种资源池(连接池,线程池等)
可以循环使用的对象