1.单例模式
概念:确保某个类只有一个实例,而且自行实例化并向整个系统提供这个实例,在计算机系统中,线程池、缓存、日志对象、对话框、打印机、显卡的驱动程序对象常被设计成单例的模式;
优点:
1.在单例模式中,活动的单例只有一个实例,对单例类的所有实例化得到的都是相同的一个实例。这样就 防止其它对象对自己的实例化,确保所有的对象都访问一个实例
2.单例模式具有一定的伸缩性,类自己来控制实例化进程,类就在改变实例化进程上有相应的伸缩性。
3.提供了对唯一实例的受控访问。
4.由于在系统内存中只存在一个对象,因此可以 节约系统资源,当 需要频繁创建和销毁的对象时单例模式无疑可以提高系统的性能。
5.允许可变数目的实例。
6.避免对共享资源的多重占用。
缺点:
1.不适用于变化的对象,如果同一类型的对象总是要在不同的用例场景发生变化,单例就会引起数据的错误,不能保存彼此的状态。
2.由于单利模式中没有抽象层,因此单例类的扩展有很大的困难。
3.单例类的职责过重,在一定程度上违背了“单一职责原则”。
4.滥用单例将带来一些负面问题,如为了节省资源将数据库连接池对象设计为的单例类,可能会导致共享连接池对象的程序过多而出现连接池溢出;如果实例化的对象长时间不被利用,系统会认为是垃圾而被回收,这将导致对象状态的丢失。
使用注意事项:
1.使用时不能用反射模式创建单例,否则会实例化一个新的对象
2.使用懒单例模式时注意线程安全问题
3.饿单例模式和懒单例模式构造方法都是私有的,因而是不能被继承的,有些单例模式可以被继承(如登记式模式)
效率:
懒汉式每次获取实例都同步,所以效率较差,其他都差不多。
①懒汉式(线程不安全)
通过提供一种静态的对象instance,利用private权限的构造方法,和静态的getInstance方法,提供给外部一个获取单例的方式
②懒汉式(线程安全)
加了一个synchronized线程锁
③懒汉式(双重校验锁)线程安全
如果有多个线程访问时,当线程A访问getInstance方法时,因为此时这个单例还没有实例化,所以他会进入synchonized锁定块这个代码区域,接下来B又访问getInstance方法,但是synchonized这个代码块已经被线程A锁定和持有了,所以线程B只能等待,当线程A成功执行单例的实例化之后,他会退出代码块,同时释放锁,此时线程B进入synchonized这个代码块,因为线程A已经实例化了,instance不为空,所以线程B就不会继续实例化这个线程
最终线程A获取到了单例的实例,并且返回,线程B没有获取到单例的实例
④懒汉式(双重校验锁+volatile)线程安全
volatile关键字:禁止指令重排,对instance的写操作有一个内存的屏障,当其他线程读取到instance==null时,会得到一个同样的当前实例的状态
⑤饿汉式(线程安全)
直接初始化instance,不管是否调用这个对象,都已经把对象创建出来了
⑥饿汉式(线程安全——静态内部类)
使用内部类,调用getInstance()方法时才加载对象达到类似懒汉式模式的效果
⑦枚举(线程安全——自动序列化)
创建枚举类型的INSTANCE,通过SingletonDemo.INSTANCE.doSomething(调用方法)
⑧threadlocal
⑨CAS锁