如何防止反射破壞單例類模式?
單例模式指出,一個類只能有一個例項,不允許建立多個例項。為此,我們將類的建構函式設為私有,並透過靜態方法返回一個例項。但是使用反射,我們仍然可以透過修改建構函式作用域來建立類的多個例項。見以下示例 -
示例 - 中斷單例
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
廣告