原文: https://howtodoinjava.com/design-patterns/creational/singleton-design-pattern-in-java/

单例模式是一种设计解决方案,其中应用希望在所有可能的情况下都具有一个且只有一个任何类的实例,而没有任何特殊情况。 在 Java 社区中,关于使任何类单例化的可能方法已经争论了很长时间。 不过,您会发现人们对您提供的任何解决方案都不满意。 它们也不能被否决。 在这篇文章中,我们将讨论一些好的方法,并将尽我们最大的努力。

  1. Table of Contents:
  2. 1\. Singleton with eager initialization
  3. 2\. Singleton with lazy initialization
  4. 3\. Singleton with static block initialization
  5. 4\. Singleton with bill pugh solution
  6. 5\. Singleton using Enum
  7. 6\. Add readResolve() to singleton objects
  8. 7\. Add serialVersionUID to singleton objects
  9. 8\. Conclusion

单例术语是从其数学对应术语中得出的。 如上所述,它希望我们每个上下文只有一个实例。 在 Java 中,每个 JVM 一个实例。

让我们看一下在 Java 中创建单例对象的可能解决方案。

1.立即初始化的单例

这是一种设计模式,其中的类实例实际上是在实际需要之前创建的。 通常,它是在系统启动时完成的。 在热切的初始化单例模式中,无论是否有其他类实际请求其实例,都将创建该单例实例。

  1. public class EagerSingleton {
  2. private static volatile EagerSingleton instance = new EagerSingleton();
  3. // private constructor
  4. private EagerSingleton() {
  5. }
  6. public static EagerSingleton getInstance() {
  7. return instance;
  8. }
  9. }

上面的方法很好用,但是有一个缺点。 不论是否在运行时都需要创建实例。 如果此实例不是大对象,并且您可以在不使用它的情况下生存下去,那么这是最好的方法。

让我们用下一种方法解决上述问题。

2.延迟初始化的单例

在计算机编程中,延迟初始化是将对象的创建,值的计算或其他昂贵的过程推迟到第一次使用时的策略。 在单例模式中,它将限制实例的创建,直到首次请求该实例为止。 让我们在代码中看到这一点:

  1. public final class LazySingleton {
  2. private static volatile LazySingleton instance = null;
  3. // private constructor
  4. private LazySingleton() {
  5. }
  6. public static LazySingleton getInstance() {
  7. if (instance == null) {
  8. synchronized (LazySingleton.class) {
  9. instance = new LazySingleton();
  10. }
  11. }
  12. return instance;
  13. }
  14. }

第一次调用时,上述方法将检查是否已使用instance变量创建了实例。 如果没有实例,即实例为null,它将创建一个实例并返回其引用。 如果实例已经创建,它将仅返回实例的引用。

但是,这种方法也有其自身的缺点。 让我们看看如何。 假设有两个线程 T1 和 T2。 两者都来创建实例并检查是否“instance==null”。 现在,两个线程都将实例变量标识为 null,因此它们都假定必须创建一个实例。 他们依次进入同步块并创建实例。 最后,我们的应用中有两个实例。

可以使用双检锁解决此错误。 该原理告诉我们在同步块中再次重新检查实例变量,如下所示:

  1. public class LazySingleton {
  2. private static volatile LazySingleton instance = null;
  3. // private constructor
  4. private LazySingleton() {
  5. }
  6. public static LazySingleton getInstance() {
  7. if (instance == null) {
  8. synchronized (LazySingleton.class) {
  9. // Double check
  10. if (instance == null) {
  11. instance = new LazySingleton();
  12. }
  13. }
  14. }
  15. return instance;
  16. }
  17. }

上面的代码是单例模式的正确实现。

请确保对实例变量使用volatile关键字,否则您可能会遇到乱码的错误情况,在实例实际构造对象之前返回实例的引用,即 JVM 仅分配了内存,而构造器代码仍未执行。 在这种情况下,引用未初始化对象的其他线程可能会引发空指针异常,甚至可能使整个应用崩溃。

3.使用静态块初始化的单例

如果您对类的加载顺序有所了解,则可以使用以下事实:即使在调用构造器之前,也要在类的加载期间执行静态块。 我们可以在单例模式中使用此功能,如下所示:

  1. public class StaticBlockSingleton {
  2. private static final StaticBlockSingleton INSTANCE;
  3. static {
  4. try {
  5. INSTANCE = new StaticBlockSingleton();
  6. } catch (Exception e) {
  7. throw new RuntimeException("Uffff, i was not expecting this!", e);
  8. }
  9. }
  10. public static StaticBlockSingleton getInstance() {
  11. return INSTANCE;
  12. }
  13. private StaticBlockSingleton() {
  14. // ...
  15. }
  16. }

上面的代码有一个缺点。 假设一个类中有 5 个静态字段,并且应用代码只需要访问 2 或 3,那么根本不需要创建实例。 因此,如果我们使用此静态初始化,尽管没有要求,我们将创建一个实例。

下一节将克服此问题。

4. 单例与 Bill Pugh 解决方案

Bill Pugh 是 java 内存模型更改背后的主要力量。 他的原则“按需初始化持有人惯例”也使用了静态块的想法,但方式有所不同。 建议使用静态内部类。

  1. public class BillPughSingleton {
  2. private BillPughSingleton() {
  3. }
  4. private static class LazyHolder {
  5. private static final BillPughSingleton INSTANCE = new BillPughSingleton();
  6. }
  7. public static BillPughSingleton getInstance() {
  8. return LazyHolder.INSTANCE;
  9. }
  10. }

如您所见,在需要实例之前,LazyHolder类直到需要时才会初始化,您仍然可以使用BillPughSingleton类的其他静态成员。 这是解决方案,我建议使用。 我在所有项目中都使用了它。

5.使用枚举的单例

这种类型的实现使用枚举。 枚举(如 java 文档中所写)为线程安全提供了隐式支持,并且仅保证了一个实例。 Java 枚举单例也是使单例花费最少的好方法。

  1. public enum EnumSingleton {
  2. INSTANCE;
  3. public void someMethod(String param) {
  4. // some class member
  5. }
  6. }

6.将readResolve()添加到单例对象

到目前为止,您必须已经决定如何实现单例。 现在,让我们看看即使在面试中也可能出现的其他问题。

假设您的应用是分布式的,并且经常将对象序列化到文件系统中,直到以后需要时才读取它们。 请注意,反序列化总是创建一个新实例。 我们来看一个例子:

我们的单例类是:

  1. public class DemoSingleton implements Serializable {
  2. private volatile static DemoSingleton instance = null;
  3. public static DemoSingleton getInstance() {
  4. if (instance == null) {
  5. instance = new DemoSingleton();
  6. }
  7. return instance;
  8. }
  9. private int i = 10;
  10. public int getI() {
  11. return i;
  12. }
  13. public void setI(int i) {
  14. this.i = i;
  15. }
  16. }

让我们对该类进行序列化,并在进行一些更改后对其进行反序列化:

  1. public class SerializationTest {
  2. static DemoSingleton instanceOne = DemoSingleton.getInstance();
  3. public static void main(String[] args) {
  4. try {
  5. // Serialize to a file
  6. ObjectOutput out = new ObjectOutputStream(new FileOutputStream(
  7. "filename.ser"));
  8. out.writeObject(instanceOne);
  9. out.close();
  10. instanceOne.setI(20);
  11. // Serialize to a file
  12. ObjectInput in = new ObjectInputStream(new FileInputStream(
  13. "filename.ser"));
  14. DemoSingleton instanceTwo = (DemoSingleton) in.readObject();
  15. in.close();
  16. System.out.println(instanceOne.getI());
  17. System.out.println(instanceTwo.getI());
  18. } catch (IOException e) {
  19. e.printStackTrace();
  20. } catch (ClassNotFoundException e) {
  21. e.printStackTrace();
  22. }
  23. }
  24. }
  25. Output:
  26. 20
  27. 10

不幸的是,两个变量的变量“i”的值不同。 显然,该类有两个实例。 因此,我们再次遇到应用中多个实例的相同问题。

要解决此问题,我们需要在DemoSingleton类中包含readResolve()方法。 当您反序列化对象时,将调用此方法。 在此方法的内部,必须返回现有实例以确保整个实例应用范围。

  1. public class DemoSingleton implements Serializable {
  2. private volatile static DemoSingleton instance = null;
  3. public static DemoSingleton getInstance() {
  4. if (instance == null) {
  5. instance = new DemoSingleton();
  6. }
  7. return instance;
  8. }
  9. protected Object readResolve() {
  10. return instance;
  11. }
  12. private int i = 10;
  13. public int getI() {
  14. return i;
  15. }
  16. public void setI(int i) {
  17. this.i = i;
  18. }
  19. }

现在,当您执行类SerializationTest时,它将为您提供正确的输出。

  1. 20
  2. 20

7.将serialVersionUId添加到单例对象

到目前为止,一切都很好。 到目前为止,我们已经解决了同步和序列化这两个问题。 现在,我们距正确而完整的实现仅一步之遥。 唯一缺少的部分是序列号。

在您的类结构在序列化和反序列化之间更改的情况下,这是必需的。 更改的类结构将导致 JVM 在反序列化过程中给出异常。

  1. java.io.InvalidClassException: singleton.DemoSingleton; local class incompatible: stream classdesc serialVersionUID = 5026910492258526905, local class serialVersionUID = 3597984220566440782
  2. at java.io.ObjectStreamClass.initNonProxy(Unknown Source)
  3. at java.io.ObjectInputStream.readNonProxyDesc(Unknown Source)
  4. at java.io.ObjectInputStream.readClassDesc(Unknown Source)
  5. at java.io.ObjectInputStream.readOrdinaryObject(Unknown Source)
  6. at java.io.ObjectInputStream.readObject0(Unknown Source)
  7. at java.io.ObjectInputStream.readObject(Unknown Source)
  8. at singleton.SerializationTest.main(SerializationTest.java:24)

仅通过向类添加唯一的串行版本 ID 即可解决此问题。 通过告诉两个类相同,这将防止编译器引发异常,并且仅加载可用的实例变量。

8.结论

在讨论了许多可能的方法和其他可能的错误情况之后,我将向您推荐以下代码模板,以设计您的单例类,该类应确保在上述所有情况下,整个应用中仅一个类的实例。

  1. public class DemoSingleton implements Serializable {
  2. private static final long serialVersionUID = 1L;
  3. private DemoSingleton() {
  4. // private constructor
  5. }
  6. private static class DemoSingletonHolder {
  7. public static final DemoSingleton INSTANCE = new DemoSingleton();
  8. }
  9. public static DemoSingleton getInstance() {
  10. return DemoSingletonHolder.INSTANCE;
  11. }
  12. protected Object readResolve() {
  13. return getInstance();
  14. }
  15. }

我希望这篇文章有足够的信息来帮助您了解单例模式单例最佳实践的最常用方法。 让我知道你的想法。

学习愉快!

实时单例示例 – 我只是想添加一些示例,以供进一步研究和在面试中提及: