Spring鬆耦合範例


物件導向的概念,是一個很好的設計來打破系統進入一個組可重用的物件。然而,當系統變大,尤其是在Java專案,龐大的物件依賴關係將一直緊密耦合引起物件難以管理或修改。在這種情況下,可以使用Spring框架作為一個核心模組輕鬆高效地管理所有的物件依賴。

輸出生成器的例子

讓我們來看一個例子,假設你的專案有一個函式輸出的內容,以CSV或JSON格式。您的程式碼可能看起來像下面的例子:

File : IOutputGenerator.java – 輸出生成器介面

package com.yiibai.output;

public interface IOutputGenerator
{
	public void generateOutput();
}

File : CsvOutputGenerator.java – 一個CSV輸出生成器用來實現IOutputGenerator介面。

package com.yiibai.output.impl;

import com.yiibai.output.IOutputGenerator;

public class CsvOutputGenerator implements IOutputGenerator
{
	public void generateOutput(){
		System.out.println("Csv Output Generator");
	}
}

File : JsonOutputGenerator.java – 一個JSON輸出生成器用來實現IOutputGenerator介面。

package com.yiibai.output.impl;

import com.yiibai.output.IOutputGenerator;

public class JsonOutputGenerator implements IOutputGenerator
{
	public void generateOutput(){
		System.out.println("Json Output Generator");
	}
}
有幾個方法來呼叫IOutputGenerator,以及如何使用 Spring 來避免物件相互結合緊密。

1. 方法1 – 直接呼叫

正常的方式,直接呼叫它。
package com.tw511.common;

import com.yiibai.output.IOutputGenerator;
import com.yiibai.output.impl.CsvOutputGenerator;

public class App 
{
    public static void main( String[] args )
    {
    	IOutputGenerator output = new CsvOutputGenerator();
    	output.generateOutput();
    }
}

存在問題

以這種方式,這個問題是「output」緊密到CsvOutputGenerator耦合,輸出生成的每一個變化可能涉及程式碼變化。如果此程式碼分散在你的專案中,輸出生成的每一次變化都會讓你受苦。

方法2 – 用輔助類呼叫它

可能想建立一個輔助類將所有輸出實現在類的內部。
package com.yiibai.output;

import com.yiibai.output.IOutputGenerator;
import com.yiibai.output.impl.CsvOutputGenerator;

public class OutputHelper
{
	IOutputGenerator outputGenerator;
	
	public OutputHelper(){
		outputGenerator = new CsvOutputGenerator();
	}
	
	public void generateOutput(){
		outputGenerator.generateOutput();
	}
	
}
通過輔助類呼叫它
package com.tw511.common;

import com.yiibai.output.OutputHelper;

public class App 
{
    public static void main( String[] args )
    {
    	OutputHelper output = new OutputHelper();
    	output.generateOutput(); 
    }
}

存在問題

這看起來比之前的更優雅,只需要管理一個輔助類,但是輔助類仍是緊耦合 CsvOutputGenerator,輸出生成的每一個變化仍然涉及小的程式碼更改。

方法3 – Spring

在這種情況下,Spring 依賴注入(DI)是一個不錯的選擇。Spring 可以讓輸出生成鬆散的耦合到輸出發生器。
OutputHelper類更小的修改。
package com.yiibai.output;

import com.yiibai.output.IOutputGenerator;

public class OutputHelper
{
	IOutputGenerator outputGenerator;
	
	public void generateOutput(){
		outputGenerator.generateOutput();
	}
	
	public void setOutputGenerator(IOutputGenerator outputGenerator){
		this.outputGenerator = outputGenerator;
	}
}
建立一個 Spring bean 的組態檔案,並在這裡宣告所有的Java物件的依賴。
<!-- Spring-Common.xml -->
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">

	<bean id="OutputHelper" class="com.yiibai.output.OutputHelper">
		<property name="outputGenerator" ref="CsvOutputGenerator" />
	</bean>
	
	<bean id="CsvOutputGenerator" class="com.yiibai.output.impl.CsvOutputGenerator" />
	<bean id="JsonOutputGenerator" class="com.yiibai.output.impl.JsonOutputGenerator" />
		
</beans>

通過Spring來呼叫它

package com.tw511.common;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

import com.yiibai.output.OutputHelper;

public class App 
{
    public static void main( String[] args )
    {
    	ApplicationContext context = 
    	   new ClassPathXmlApplicationContext(new String[] {"Spring-Common.xml"});

    	OutputHelper output = (OutputHelper)context.getBean("OutputHelper");
    	output.generateOutput();
    	  
    }
}
現在,只需要改變 Spring XML 檔案使用不同的輸出生成器。只修改 Spring XML 檔案而不需要無碼修改,這意味著更少的錯誤。

結論

有了Spring框架 - 這種依賴注入(DI)為物件的依賴關係管理有用的特性,使大型Java專案開發管理中更優雅的,高度靈活和便於維護。