如何防止反射破壞單例類模式?


單例模式指出,一個類只能有一個例項,不允許建立多個例項。為此,我們將類的建構函式設為私有,並透過靜態方法返回一個例項。但是使用反射,我們仍然可以透過修改建構函式作用域來建立類的多個例項。見以下示例 -

示例 - 中斷單例

 實際演示

import java.io.Serializable;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;

public class Tester {
   public static void main(String[] args) throws
   InstantiationException, IllegalAccessException,

   IllegalArgumentException, InvocationTargetException{
      A a = A.getInstance();
      A b = null;

      Constructor<?>[] constructors = A.class.getDeclaredConstructors();

      for (Constructor constructor : constructors) {
         //make the private constructor as public
         constructor.setAccessible(true);
         b = (A) constructor.newInstance();
         break;
      }
      System.out.println(a.hashCode());
      System.out.println(b.hashCode());
   }
}

class A implements Serializable {
   private static A a;
   private A(){}

   public static A getInstance(){
      if(a == null){
         a = new A();
      }
      return a;
   }
}

輸出

705927765
366712642

這裡你可以看到,我們建立了一個單例類的另一個物件。讓我們看看如何防止這種情況 -

建立 A 使用列舉而不是類。

示例 - 保護單例

 實際演示

import java.io.Serializable;
import java.lang.reflect.InvocationTargetException;

public class Tester {
   public static void main(String[] args) throws
   InstantiationException, IllegalAccessException,
   IllegalArgumentException, InvocationTargetException{

      A a = A.INSTANCE;
      A b = A.INSTANCE;

      System.out.println(a.hashCode());
      System.out.println(b.hashCode());
   }
}
enum A implements Serializable {
   INSTANCE;
}

輸出

705927765
705927765

更新於: 23-Jun-2020

746 次瀏覽

開啟 職業生涯

完成課程,獲得認證

開始吧
廣告