之前在java基础的笔记里有写过部分的单例(单例设计模式概述),这是基础的懒汉式饿汉式,今天写的这个主要也是对这个知识点归纳在拓展。
其实很意外的,想着想着搜着搜着就看到了那么种第一眼就觉得惊艳的单例,这次着重介绍一下。先来小菜:
饿汉式
class Singleton{ private static Singleton s = new Singleton(); private hungry(){} public static Singleton getInstance(){ return s; } public void print(){ System.out.println("饿汉式"); } }
懒汉式,普通的那种
class Singleton{ private static Singleton i = null; private Singleton(){} public static Singleton getInstance(){ if(i == null){ i = new Singleton(); } return i; } public void print(){ System.out.println("懶漢式"); } }
懒汉式,同步方法
class Singleton{ private static Singleton i = null; private Singleton(){} public synchronized static Singleton getInstance(){ if(i == null){ i = new Singleton(); } return i; } public void print(){ System.out.println("懶漢式同步方法"); } }
懒汉式,同步代码块
class Singleton{ private static Singletoni = null; private Singleton(){} public static Singleton getInstance(){ if(i == null){ synchronized(Singleton.class){ if(i == null){ i = new Singleton(); } } } return i; } public void print(){ System.out.println("懶漢式同步代碼塊"); } }
到这里顺便提一句,这个强行加上的方式是有意义的,同步锁有所谓的等级,从他能够限制的层次来分这个级别,最低一级是同步方法,方法锁,只在本方法内同步,高一级是同步代码块,且内部以对象为锁,我叫他对象锁,只要是这个对象调用此方法都具有同步效应,最高一级是类锁,同步代码块中以字节码文件为锁对象,即属于该类的对象调用此方法都将具有同步效果。
暂且叫直接式
class Singleton{ private Singleton(){} public static final Singleton s = new Singleton(); public void print(){ System.out.println("直接式"); } }
直接将该对象使用public static final修饰,而构造方法依旧是私有的,以类直接调用该对象即可。
重点介绍,我叫他高级姿势(逃
public class Singleton { private Singleton() {} private static class SingletonHolder { static Singleton instance = new Singleton(); } public static Singleton getInstance() { return SingletonHolder.instance; } }
既然是重点介绍,就从各个角度来进行了
- 和饿汉式相比较来说,他延迟加载并且以静态的方式实现了对象的同步。
- 和懒汉式相比较,不需要再使用synchronized关键字重复判断,也能够实现同步,并且同时具有延迟加载的的功能。
- 好吧,我承认并不具体,我讲不出些什么了Σ( ° △ °|||)︴
但这并不能影响他确实很高级,以静态的内部类实现这样的单例,确实,有没有很惊艳?!
顺带提一句,apache的源码里大多数采用的就是这种类型的单例。