1.恶汉1号
估计以前经常这么写,现在很少这么写,因为有更好的模式
2.恶汉2号
跟饿汉1号类似。还是在程序启动就生成了
3.懒汉1号
多线程有事。单线程没事
4.懒汉2号
虽然加了sync标识,但是还是无法保证多线程没事
5.懒汉3号双重检测
同步代码中再加一个 if else 。同时加上volatile 防止 singleton=new singleton();这一句指令重排序。安全
singleton=new singleton(); 其实是三句指令 第一句a为新对象分配一个地址 后两句 生成对象b 把地址赋值给singleton变量c 如果是先 a,b,c顺序没问题。如果是acb。 那singleton可能拿到了一个空对象。
6.内部静态类
跟饿汉模式有点类似,只是在类装载的时候,SingletonInstance并没有被主动使用,只有显式调用getInstance方法是,才会调用装载SingletoInstance类,从而实例化instance,既实现了线程安全,又避免了同步带来的性能影响。
这里普通的内部静态类跟懒汉双重检测这种模式 这两种模式都不能防止通过反射方式再构造出一个对象。
需要添加一些额外代码,比如防止反射后再构造一个对象。在构造方法增加一个计数器检查,超了就抛出异常,
比如为了保证序列还是同一个,重载方法 readResolve() 返回单例对象,但是这些在枚举类里头都是默认实现了。
7.枚举类单例用更少的代码实现单例,并且防止反射后再构造单例。并且还保证了序列化之后还是同一个。