演進式例解控制反轉(IoC)、依賴注入(DI)之一

原文地址:http://haolloyin.blog.51cto.com/1177454/458416/html


近來老是接觸到 IoCInversion of Control,控制反轉)、DIDependency Injection,依賴注入)等編程原則或者模式,而這些是著名 Java 框架 SpringStruts 等的核心所在。針對此查了Wikipedia 中各個條目,並從圖書館借來相關書籍,閱讀後有些理解,現結合書中的講解以及本身的加工整理以下:java

 
問題描述:

開發一個可以按照不一樣要求生成Excel PDF 格式的報表的系統,例如日報表、月報表等等。編程

 
解決方案:

根據「面向接口編程」的原則,應該分離接口與實現,即將生成報表的功能提取爲一個通用接口ReportGenerator,並提供生成 Excel  PDF格式報表的兩個實現類 ExcelGenerator PDFGenerator,而客戶Client 再經過服務提供者 ReportService 獲取相應的報表打印功能。框架

 
實現方法:
根據上面所述,獲得以下類圖:

 
代碼實現:
 
 
  1. interface ReportGenerator { 
  2.     public void generate(Table table); 
  3.  
  4. class ExcelGenerator implements ReportGenerator { 
  5.     public void generate(Table table) { 
  6.         System.out.println("generate an Excel report ..."); 
  7.     } 
  8.  
  9. class PDFGenerator implements ReportGenerator { 
  10.     public void generate(Table table) { 
  11.         System.out.println("generate an PDF report ..."); 
  12.     } 
  13.  
  14. class ReportService { 
  15.     // 負責建立具體須要的報表生成器 
  16.     private ReportGenerator generator = new PDFGenerator()
  17.     // private static ReportGenerator generator = new ExcelGenerator(); 
  18.      
  19.     public void getDailyReport(Date date) { 
  20.         table.setDate(date); 
  21.         // ... 
  22.         generator.generate(table); 
  23.     } 
  24.      
  25.     public void getMonthlyReport(Month month) { 
  26.         table.setMonth(month); 
  27.         // ... 
  28.         generator.generate(table); 
  29.     } 
  30.  
  31.  
  32. public class Client { 
  33.     public static void main(String[] args) { 
  34.         ReportService reportService = new ReportService(); 
  35.         reportService.getDailyReport(new Date()); 
  36.         //reportService.getMonthlyReport(new Date()); 
  37.     } 
 
 
問題描述:

如上面代碼中的註釋所示,具體的報表生成器由 ReportService 類內部硬編碼建立,由此ReportService 已經直接依賴 PDFGenerator  ExcelGenerator ,必須消除這一明顯的緊耦合關係。學習

 
解決方案: 引入容器

引入一箇中間管理者,也就是容器(Container),由其統一管理報表系統所涉及的對象(在這裏是組件,咱們將其稱爲 Bean),包括 ReportService 和各個 XXGenerator 。在這裏使用一個鍵-值對形式的 HashMap 實例來保存這些 Bean編碼

 
實現方法:
獲得類圖以下:

 
代碼實現:
 
  1. class Container { 
  2.     // 以鍵-值對形式保存各類所需組件 Bean 
  3.     private static Map<String, Object> beans
  4.      
  5.     public Container() { 
  6.         beans = new HashMap<String, Object>(); 
  7.          
  8.         // 建立、保存具體的報表生起器 
  9.         ReportGenerator reportGenerator = new PDFGenerator()
  10.         beans.put("reportGenerator", reportGenerator); 
  11.          
  12.         // 獲取、管理 ReportService 的引用 
  13.         ReportService reportService = new ReportService(); 
  14.         beans.put("reportService", reportService); 
  15.     } 
  16.      
  17.     public static Object getBean(String id) { 
  18.         return beans.get(id); 
  19.     } 
  20.  
  21. class ReportService { 
  22.     // 消除緊耦合關係,由容器取而代之 
  23.     // private static ReportGenerator generator = new PDFGenerator(); 
  24.     private ReportGenerator generator = (ReportGenerator) Container.getBean("reportGenerator"); 
  25.  
  26.     public void getDailyReport(Date date) { 
  27.         table.setDate(date); 
  28.         generator.generate(table); 
  29.     } 
  30.      
  31.     public void getMonthlyReport(Month month) { 
  32.         table.setMonth(month); 
  33.         generator.generate(table); 
  34.     } 
  35.  
  36. public class Client { 
  37.     public static void main(String[] args) { 
  38.         Container container = new Container(); 
  39.         ReportService reportService = (ReportService)Container.getBean("reportService"); 
  40.         reportService.getDailyReport(new Date()); 
  41.         //reportService.getMonthlyReport(new Date()); 
  42.     } 
 
時序圖大體以下:

 
效果:

如上面所示,ReportService 再也不與具體的 ReportGenerator 直接關聯,已經用容器將接口和實現隔離開來了,提升了系統組件 Bean 的重用性,此時還可使用配置文件在 Container 中實時獲取具體組件的定義。spa

 
 
問題描述:

然而,觀察上面的類圖,很容易發現 ReportService  Container 之間存在雙向關聯,彼此互相有依賴關係。而且,若是想要重用 ReportService,因爲它也是直接依賴於單獨一個 Container 具體查找邏輯。若其餘容器具體不一樣的組件查找機制(如 JNDI),此時重用 ReportService 意味着須要修改 Container 的內部查找邏輯。設計

 
解決方案: 引入  Service Locator

再次引入一個間接層 Service Locator,用於提供組件查找邏輯的接口,請看Wikipedia 中的描述 或者 Java EE 對其的描述1 描述2 。這樣就可以將可能變化的點隔離開來。code

 
實現方法:
類圖以下:

 
代碼實現:
 
 
  1. // 實際應用中能夠是用 interface 來提供統一接口 
  2. class ServiceLocator { 
  3.     private static Container container = new Container(); 
  4.      
  5.     public static ReportGenerator getReportGenerator() { 
  6.         return (ReportGenerator)container.getBean("reportGeneraator"); 
  7.     } 
  8.  
  9. class ReportService { 
  10.     private ReportGenerator reportGenerator = ServiceLocator.getReportGenerator(); 
  11.      
  12.     // ... 


 

小結:htm

一、雖然講了這麼大篇幅尚未進入真正的主題——IoCDI,不過已經在一步步逼近了,下一篇應該會更精彩!在這裏...

二、能夠很明顯地看得出上面兩中從新設計以解耦、隔離變化點都是經過引入間接層得以解決的。

三、在看書過程當中,我感受《Spring 攻略》一書中以「問題描述、解決方案、實現方法」方式的講解比較容易理解和理清思路,故而也學習用這種方式來寫。另,推薦該書以學習 Spring 框架(儘管目前我看得也很少)。

相關文章
相關標籤/搜索