Abstract Factory 是另一種被普遍運用的創建型模式,Abstract Factory 通過專門的 Factory Class 來封裝對象創建的職責,并通過實現 Abstract Factory 來完成不同的創建邏輯。如果被測試單元所使用的外部對象是通過 Abstract Factory 創建的,則實現一個新的 Concrete Factory,并在此 Factory 中創建 Mock Objects 是一個比較好的解決辦法。對于 Factory 本身,則可以在 setUp 測試的時候指定新的 Concrete Factory ;此外,借助依賴注入框架(如 Spring 的 BeanFactory),通過依賴注入的方式將 Factory 注入也是一種不錯的解決方案。對于簡單的依賴注入需求,可以考慮實現一個應用專有的依賴注入模塊,或者實現一個簡單的實現加載器,即根據配置文件載入相應的實現,從而無需修改應用代碼,僅通過修改配置文件即可載入不同的實現,進而方便地修改程序的運行路徑,執行單元測試。
下面的代碼實現了一個簡單的 InstanceFactory:
// refer to http://www.opensc-project.org/opensc-java/export/100/trunk/ // pkcs15/src/main/java/org/opensc/pkcs15/asn1/InstanceFactory.java packagecom.instancefactory.demo; importjava.lang.reflect.InvocationTargetException; importjava.lang.reflect.Method; importjava.lang.reflect.Modifier; public class InstanceFactory { private final Method getInstanceMethod; public InstanceFactory(String type) { Class clazz =null; try { clazz = Class.forName(type); this.getInstanceMethod = clazz.getMethod("getInstance"); if(!Modifier.isStatic(this.getInstanceMethod.getModifiers()) || !Modifier.isPublic(this.getInstanceMethod.getModifiers())) throw new IllegalArgumentException( "Method [" + clazz.getName() + ".getInstance(Object)] is not static public."); } catch (NoSuchMethodException e) { throw new IllegalArgumentException( "Class [" + clazz.getName() + "] has no static getInstance(Object) method.", e); } catch (ClassNotFoundException e) { throw new IllegalArgumentException("Class [" + type + "] is not found"); } } |