簡介
面向切面編程(AOP)提供另外一種角度來思考程序結構,通過這種方式彌補了面向對象編程(OOP)的不足。 除了類(classes)以外,AOP提供了切面。切面對關注點進行模塊化,例如橫切多個類型和對象的事務管理。 (這些關注點術語通常稱作橫切(crosscutting) 關注點。)
Spring的一個關鍵的組件就是AOP框架。 儘管如此,Spring IoC容器並不依賴於AOP,這意味著你可以自由選擇是否使用AOP,AOP提供強大的中間件解決方案,這使得Spring IoC容器更加完善。
Spring 2.0 AOP:
Spring 2.0 引入了一種更加簡單並且更強大的方式來自定義切面,用戶可以選擇使用基於模式(schema-based)的方式或者使用@AspectJ註解。 對於新的應用程序,如果用戶使用Java 5開發,我們推薦用戶使用@AspectJ風格,否則可以使用基於模式的風格。 這兩種風格都完全支持通知(Advice)類型和AspectJ的切入點語言,雖然實際上仍然使用Spring AOP進行織入(Weaving)。
本章主要討論Spring 2.0對基於模式和基於@AspectJ的AOP支持。 Spring 2.0完全保留了對Spring 1.2的向下兼容性,下一章將討論Spring 1.2 API所提供的底層的AOP支持。
Spring中所使用的AOP:
提供聲明式企業服務,特別是為了替代EJB聲明式服務。 最重要的服務是聲明性事務管理(declarative transaction management) , 這個服務建立在Spring的抽象事務管理(transaction abstraction)之上。
允許用戶實現自定義的切面,用AOP來完善OOP的使用。
實例
我們經常會用到的有如下幾種
1、基於代理的AOP
2、純簡單java對象切面
3、@Aspect註解形式的
4、注入形式的Aspcet切面下面我們就一個一個來應用吧.
下面先寫一下幾個基本的類。
接口類:
/** * 定義一個接口*/ public interface Sleepable { /** * 睡覺方法*/ void sleep(); }實現類:
/** * 本人實現睡覺接口*/ public class ChenLliNa implements Sleepable { @Override public void sleep() { // TODO Auto-generated method stub System.out.println("乖,該睡覺了!"); } }增強類:
/** * 定義一個睡眠的增強同時實現前置和後置*/ public class SleepHelper implements MethodBeforeAdvice, AfterReturningAdvice { @Override public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable { System.out.println("睡覺前要敷面膜"); } @Override public void before(Method method, Object[] args, Object target) throws Throwable { System.out.println("睡覺後要做美夢"); } }一、基於代理的AOP
<!-- 創建一個增強advice --> <bean id ="sleepHelper"/> <bean id="lina"/> <!-- 定義切點匹配所有的sleep方法--> <bean id ="sleepPointcut"> <property name="pattern" value=".*sleep"></property> </bean> <!-- 切面增強+切點結合--> <bean id="sleepHelperAdvisor"> <property name="advice" ref="sleepHelper"/> <property name="pointcut" ref="sleepPointcut"/> </bean> <!-- 定義代理對象--> <bean id="linaProxy"> <property name="target" ref="lina"/> <property name="interceptorNames" value="sleepHelperAdvisor"/> <!-- <property name="proxyInterfaces" value="com.tgb.springaop.service.Sleepable"/> --> </bean>
如配置文件中:
pattern屬性指定了正則表達式,他匹配所有的sleep方法使用org.springframework.aop.support.DefaultPointcutAdvisor的目的是為了使切點和增強結合起來形成一個完整的切面最後配置完後通過org.springframework.aop.framework.ProxyFactoryBean產生一個最終的代理對象。
二、純簡單java對象切面
純簡單java對象切面這話怎麼說呢,在我看來就是相對於第一種配置,不需要使用代理,,而是通過spring的內部機制去自動掃描,這時候我們的配置文件就該如下修改:
<!-- 創建一個增強advice --> <bean id ="sleepHelper"/> <!-- 目標類--> <bean id="lina"/> <!-- 配置切點和通知--> <bean id ="sleepAdvisor"> <property name="advice" ref="sleepHelper"></property> <property name="pattern" value=".*sleep"/> </bean> <!-- 自動代理配置--> <bean/>
是不是相對於第一種簡單了許多,不用再去配置代理了。
三、@Aspect註解形式
根據我們的經驗也知道,註解的形式相對於配置文件是簡單一些的,這時候需要在已有的方法或類上家註解:
/** * 通過註解的方式添加增強*/ @Aspect @Component public class SleepHelper03 { /*@Pointcut("execution(* com.tgb.springaop.service.impl..*(..))")*/ @Pointcut("execution(* *.sleep(..))") public void sleeppoint(){} @Before("sleeppoint()") public void beforeSleep(){ System.out.println("睡覺前要敷面膜"); } @AfterReturning("sleeppoint()") public void afterSleep(){ System.out.println("睡覺後要做美夢"); }
配置文件中只需寫:
<!--掃描包--> <context:component-scan base-package="com.tgb" annotation-config="true"/> <!-- ASPECTJ註解--> <aop:aspectj-autoproxy proxy-target-class="true" /> <!-- 目標類--> <bean id="lina"/>
四、注入形式的Aspcet切面
個人感覺這個是最簡單的也是最常用的,也是最靈活的。配置文件如下:
<!-- 目標類--> <bean id="lina"/> <bean id ="sleepHelper"/> <aop:config> <aop:aspect ref="sleepHelper"> <aop:before method="beforeSleep" pointcut="execution(* *.sleep(..))"/> <aop:after method="afterSleep" pointcut="execution(* *.sleep(..))"/> </aop:aspect> </aop:config>
配置文件中提到的SleepHelper02類如下:
/** * 通過註解的方式添加增強*/ public class SleepHelper02 { public void beforeSleep(){ System.out.println("睡覺前要敷面膜"); } public void afterSleep(){ System.out.println("睡覺後要做美夢"); } }
是不是看上去都很簡單呀,這樣是不是大家都會使用spring aop了? !
關於如何調用,這裡寫了幾個測試類,可以看一下,基本都一樣:
/** * 配置文件spring_aop.xml 通過代理*/ @Test public void test(){ ApplicationContext ct = new ClassPathXmlApplicationContext("spring_aop.xml"); Sleepable sleeper =(Sleepable) ct.getBean("linaProxy"); sleeper.sleep(); } /** * 配置文件spring_aop_01.xml 簡答的java對象*/ @Test public void test01(){ ApplicationContext ct = new ClassPathXmlApplicationContext("spring_aop_01.xml"); Sleepable sleeper = (Sleepable)ct.getBean("lina"); sleeper.sleep(); } /** * 配置文件spring_aop_03.xml 通過aspect註解*/ @Test public void test03(){ ApplicationContext ct = new ClassPathXmlApplicationContext("spring_aop_03.xml"); Sleepable sleeper = (Sleepable)ct.getBean("lina"); sleeper.sleep(); } /** * 配置文件spring_aop_02.xml 通過apsect配置文件* @author 陳麗娜* @version 2015年5月31日上午10:09:37 */ @Test public void test02(){ ApplicationContext ct = new ClassPathXmlApplicationContext("spring_aop_02.xml"); Sleepable sleeper = (Sleepable)ct.getBean("lina"); sleeper.sleep(); }
通過測試類可以看出,不管以什麼樣的方式來實現aop他們的使用都是沒有差別的,這幾個測試類的結果都是一樣的: