什么是单例?

单例(Singleton)对象的类必须保证只有一个实例存在——这是维基百科上对单例的定义。
对单例的实现可以分为两大类——懒汉式和饿汉式。

懒汉式和饿汉式的区别:

懒汉式:指全局的单例实例在第一次被使用是构建。
饿汉式:指全局的单例实例再类装载是构建。
懒汉式单例:
关于JAVA的单例
上述单例有关的知识点:
把构造器改称为私有的,这样能够防止被外部的类调用。
加上synchronized关键字后,getSingle方法会锁定。如果有两个线程,(T1,T2)同时执行这个方法时,会有其中一个线程T1获得同步锁,得以继续执行,而另一个线程T2则需要等待,当T1执行完毕getSingle之后(完成了Null判断,对象创建,或得返回值之后),T2线程才会执行。
有两次if(single== null)的判断,这个叫做双重检查【Double-Check】。
什么是原子操作:
原子操作(atomic)就是不可分割的操作,在计算机中,就是指不会因为线程调动而打断的操作。
什么是指重排:
简单来说,就是计算机为了提高执行效率,会做的一些优化,在不影响最终的情况下,可能会对一些语句的执行顺序做一些调整。
volatile关键字的一个作用是禁止指重排。

饿汉式单例:

关于JAVA的单例对于饿汉式单例来,基本是完美的。
使用他的缺点就只是饿汉式单例本身的存在了 ——由于single的初始化是在类加载时进行的,而类的加载是由ClassLoader来做的,所以开发者对于他的初始化的时机会很难去把握:

  1. 可能由于初始化的太早,造成资源的浪费.
  2. 如果初始化本身依赖于一些其他数据,那么也就很难保证其他数据会在它初始化之前准备好。 当然,如果所需的单例占用的资源很少,并且也不依赖于其他数据,那么这种实现方式也是很好的。
    什么时候是类装载时?
    大致有这几个条件时,会触发一个类别加载:
    1.new 一个对象时。
    2.使用反射创建他的实例时。
    3.子类被加载时,如果父类没被加载,就先加载父类。
    4.jvm启动时执行的主类会首先被加载。

一些其他的实现方式:

1.Effective java1-静态内部类
关于JAVA的单例
2.Effective java-2 枚举:

public enum SingleD {
    SINGLE_D;
    public void getSingD(){
        //do something
    }
}
//使用
SingleD.SINGLE_D.getSingD();

这是一个枚举类型,连class都没用。
这种写法在功能上与共有域方法相近,但他更简洁,无偿的提供了序列化机制,绝对防止对此实例化,及时在面对复杂得序列化和反射攻击的时候。

相关文章:

  • 2022-12-23
  • 2022-12-23
  • 2021-08-22
  • 2022-12-23
  • 2021-11-07
  • 2022-01-31
  • 2021-07-30
  • 2021-11-21
猜你喜欢
  • 2021-10-29
  • 2021-10-24
  • 2021-11-03
  • 2021-06-11
  • 2021-04-14
  • 2021-09-25
  • 2022-12-23
相关资源
相似解决方案