作者:Xuefeng 文章來源:本站原創(chuàng) 點擊數(shù):656 更新時間:2006-4-2
提示:如果您有任何疑問,請到
開發(fā)論壇提問或討論。
Spring是一個非常優(yōu)秀的輕量級框架,通過Spring的IoC容器,我們的關注點便放到了需要實現(xiàn)的業(yè)務邏輯上。對AOP的支持則能讓我們動態(tài)增強業(yè)務方法。編寫普通的業(yè)務邏輯Bean是非常容易而且易于測試的,因為它能脫離J2EE容器(如Servlet,JSP環(huán)境)單獨進行單元測試。最后的一步便是在Spring框架中將這些業(yè)務Bean以XML配置文件的方式組織起來,它們就按照我們預定的目標正常工作了!非常容易!
本文將給出一個基本的Spring入門示例,并演示如何使用Spring的AOP將復雜的業(yè)務邏輯分離到每個方面中。
1.開發(fā)環(huán)境配置
2.編寫B(tài)ean接口及其實現(xiàn)
3.在Spring中配置Bean并獲得Bean的實例
4.編寫Advisor以增強ServiceBean
5.總結
1.開發(fā)環(huán)境配置
首先,需要正確配置Java環(huán)境。推薦安裝JDK1.4.2,并正確配置環(huán)境變量:
JAVA_HOME=<JDK安裝目錄>
CLASSPATH=.
Path=%JAVA_HOME%\bin;……
我們將使用免費的Eclipse 3.1作為IDE。新建一個Java Project,將Spring的發(fā)布包spring.jar以及commons-logging-1.0.4.jar復制到Project目錄下,并在Project > Properties中配置好Java Build Path:
2.編寫B(tài)ean接口及其實現(xiàn)
我們實現(xiàn)一個管理用戶的業(yè)務Bean。首先定義一個ServiceBean接口,聲明一些業(yè)務方法:
/**
* Copyright_2006, Liao Xuefeng
* Created on 2006-3-9
* For more information, please visit:
http://www.crackj2ee.com*/
package com.crackj2ee.example.spring; /**
* Interface of service facade.
*
* @author Xuefeng
*/
public interface ServiceBean {
void addUser(String username, String password);
void deleteUser(String username);
boolean findUser(String username);
String getPassword(String username);
}
然后在MyServiceBean中實現(xiàn)接口:
/**
* Copyright_2006, Liao Xuefeng
* Created on 2006-3-9
*
* For more information, please visit:
http://www.crackj2ee.com*/
package com.crackj2ee.example.spring; import java.util.*; public class MyServiceBean implements ServiceBean { private String dir;
private Map map = new HashMap(); public void setUserDir(String dir) {
this.dir = dir;
System.out.println("Set user dir to: " + dir);
} public void addUser(String username, String password) {
if(!map.containsKey(username))
map.put(username, password);
else
throw new RuntimeException("User already exist.");
} public void deleteUser(String username) {
if(map.remove(username)==null)
throw new RuntimeException("User not exist.");
} public boolean findUser(String username) {
return map.containsKey(username);
} public String getPassword(String username) {
return (String)map.get(username);
}
}
為了簡化邏輯,我們使用一個Map保存用戶名和口令。
現(xiàn)在,我們已經(jīng)有了一個業(yè)務Bean。要測試它非常容易,因為到目前為止,我們還沒有涉及到Spring容器,也沒有涉及到任何Web容器(假定這是一個Web應用程序關于用戶管理的業(yè)務Bean)。完全可以直接進行Unit測試,或者,簡單地寫個main方法測試:
/**
* Copyright_2006, Liao Xuefeng
* Created on 2006-3-9
* For more information, please visit:
http://www.crackj2ee.com*/
package com.crackj2ee.example.spring; public class Main { public static void main(String[] args) throws Exception {
ServiceBean service = new MyServiceBean();
service.addUser("bill", "hello");
service.addUser("tom", "goodbye");
service.addUser("tracy", "morning");
System.out.println("tom‘s password is: " + service.getPassword("tom"));
if(service.findUser("tom")) {
service.deleteUser("tom");
}
}
}
執(zhí)行結果:
3.在Spring中配置Bean并獲得Bean的實例
我們已經(jīng)在一個main方法中實現(xiàn)了業(yè)務,不過,將對象的生命周期交給容器管理是更好的辦法,我們就不必為初始化對象和銷毀對象進行硬編碼,從而獲得更大的靈活性和可測試性。
想要把ServiceBean交給Spring來管理,我們需要一個XML配置文件。新建一個beans.xml,放到src目錄下,確保在classpath中能找到此配置文件,輸入以下內容:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN"
"
<beans>
<bean id="service" class="com.crackj2ee.example.spring.MyServiceBean" />
</beans>
以上XML聲明了一個id為service的Bean,默認地,Spring為每個聲明的Bean僅創(chuàng)建一個實例,并通過id來引用這個Bean。下面,我們修改main方法,讓Spring來管理業(yè)務Bean:
/**
* Copyright_2006, Liao Xuefeng
* Created on 2006-3-9
* For more information, please visit:
http://www.crackj2ee.com*/
package com.crackj2ee.example.spring; import org.springframework.beans.factory.xml.XmlBeanFactory;
import org.springframework.core.io.ClassPathResource; public class Main { public static void main(String[] args) throws Exception {
// init factory:
XmlBeanFactory factory = new XmlBeanFactory(new ClassPathResource("beans.xml"));
// use service bean:
ServiceBean service = (ServiceBean)factory.getBean("service");
service.addUser("bill", "hello");
service.addUser("tom", "goodbye");
service.addUser("tracy", "morning");
System.out.println("tom‘s password is \"" + service.getPassword("tom") + "\"");
if(service.findUser("tom")) {
service.deleteUser("tom");
}
// close factory:
factory.destroySingletons();
}
}
執(zhí)行結果:
由于我們要通過main方法啟動Spring環(huán)境,因此,首先需要初始化一個BeanFactory。紅色部分是初始化Spring的BeanFactory的典型代碼,只需要保證beans.xml文件位于classpath中。
然后,在BeanFactory中通過id查找,即可獲得相應的Bean的實例,并將其適當轉型為合適的接口。
接著,實現(xiàn)一系列業(yè)務操作,在應用程序結束前,讓Spring銷毀所有的Bean實例。
對比上一個版本的Main,可以看出,最大的變化是不需要自己管理Bean的生命周期。另一個好處是在不更改實現(xiàn)類的前提下,動態(tài)地為應用程序增加功能。
4.編寫Advisor以增強ServiceBean
所謂AOP即是將分散在各個方法處的公共代碼提取到一處,并通過類似攔截器的機制實現(xiàn)代碼的動態(tài)織入??梢院唵蔚叵胂蟪?,在某個方法的調用前、返回前、調用后和拋出異常時,動態(tài)插入自己的代碼。在弄清楚Pointcut、Advice之類的術語前,不如編寫一個最簡單的AOP應用來體驗一下。
考慮一下通常的Web應用程序都會有日志記錄,我們來編寫一個LogAdvisor,對每個業(yè)務方法調用前都作一個記錄:
/**
* Copyright_2006, Liao Xuefeng
* Created on 2006-3-9
* For more information, please visit:
http://www.crackj2ee.com*/
package com.crackj2ee.example.spring; import java.lang.reflect.Method;
import org.springframework.aop.MethodBeforeAdvice; public class LogAdvisor implements MethodBeforeAdvice {
public void before(Method m, Object[] args, Object target) throws Throwable {
System.out.println("[Log] " + target.getClass().getName() + "." + m.getName() + "()");
}
}
然后,修改beans.xml:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN"
"
<bean id="serviceTarget" class="com.crackj2ee.example.spring.MyServiceBean" /> <bean id="logAdvisor" class="com.crackj2ee.example.spring.LogAdvisor" /> <bean id="service" class="org.springframework.aop.framework.ProxyFactoryBean">
<property name="proxyInterfaces"><value>com.crackj2ee.example.spring.ServiceBean</value></property>
注意觀察修改后的配置文件,我們使用了一個ProxyFactoryBean作為service來與客戶端打交道,而真正的業(yè)務Bean即MyServiceBean被聲明為serviceTarget并作為參數(shù)對象傳遞給ProxyFactoryBean,proxyInterfaces指定了返回的接口類型。對于客戶端而言,將感覺不出任何變化,但卻動態(tài)加入了LogAdvisor,關系如下:
package com.crackj2ee.example.spring; import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation; public class PasswordAdvisor implements MethodInterceptor {
這個PasswordAdvisor將截獲ServiceBean的getPassword()方法的返回值,并將其改為"***"。繼續(xù)修改beans.xml: