簡(jiǎn)介
面向切面編程(AOP)提供另外一種角度來(lái)思考程序結(jié)構(gòu),通過(guò)這種方式彌補(bǔ)了面向?qū)ο缶幊蹋∣OP)的不足。 除了類(lèi)(classes)以外,AOP提供了 切面。切面對(duì)關(guān)注點(diǎn)進(jìn)行模塊化,例如橫切多個(gè)類(lèi)型和對(duì)象的事務(wù)管理。 (這些關(guān)注點(diǎn)術(shù)語(yǔ)通常稱(chēng)作 橫切(crosscutting) 關(guān)注點(diǎn)。)
Spring的一個(gè)關(guān)鍵的組件就是 AOP框架。 盡管如此,Spring IoC容器并不依賴(lài)于AOP,這意味著你可以自由選擇是否使用AOP,AOP提供強(qiáng)大的中間件解決方案,這使得Spring IoC容器更加完善。
Spring 2.0 AOP:
Spring 2.0 引入了一種更加簡(jiǎn)單并且更強(qiáng)大的方式來(lái)自定義切面,用戶(hù)可以選擇使用基于模式(schema-based)的方式或者使用@AspectJ注解。 對(duì)于新的應(yīng)用程序,如果用戶(hù)使用Java 5開(kāi)發(fā),我們推薦用戶(hù)使用@AspectJ風(fēng)格,否則可以使用基于模式的風(fēng)格。 這兩種風(fēng)格都完全支持通知(Advice)類(lèi)型和AspectJ的切入點(diǎn)語(yǔ)言,雖然實(shí)際上仍然使用Spring AOP進(jìn)行織入(Weaving)。
本章主要討論Spring 2.0對(duì)基于模式和基于@AspectJ的AOP支持。 Spring 2.0完全保留了對(duì)Spring 1.2的向下兼容性,下一章 將討論Spring 1.2 API所提供的底層的AOP支持。
Spring中所使用的AOP:
提供聲明式企業(yè)服務(wù),特別是為了替代EJB聲明式服務(wù)。 最重要的服務(wù)是 聲明性事務(wù)管理(declarative transaction management) , 這個(gè)服務(wù)建立在Spring的抽象事務(wù)管理(transaction abstraction)之上。
允許用戶(hù)實(shí)現(xiàn)自定義的切面,用AOP來(lái)完善OOP的使用。
實(shí)例
我們經(jīng)常會(huì)用到的有如下幾種
1、基于代理的AOP
2、純簡(jiǎn)單java對(duì)象切面
3、@Aspect注解形式的
4、注入形式的Aspcet切面
下面我們就一個(gè)一個(gè)來(lái)應(yīng)用吧.
下面先寫(xiě)一下幾個(gè)基本的類(lèi)。
接口類(lèi):
1
2
3
4
5
6
7
8
9
10
|
/** * 定義一個(gè)接口 */ public interface Sleepable { /** * 睡覺(jué)方法 */ void sleep(); } |
實(shí)現(xiàn)類(lèi):
1
2
3
4
5
6
7
8
9
10
11
|
/** * 本人實(shí)現(xiàn)睡覺(jué)接口 */ public class ChenLliNa implements Sleepable { @Override public void sleep() { // TODO Auto-generated method stub System.out.println( "乖,該睡覺(jué)了!" ); } } |
增強(qiáng)類(lèi):
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
/** * 定義一個(gè)睡眠的增強(qiáng) 同時(shí)實(shí)現(xiàn)前置 和后置 */ public class SleepHelper implements MethodBeforeAdvice, AfterReturningAdvice { @Override public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable { System.out.println( "睡覺(jué)前要敷面膜" ); } @Override public void before(Method method, Object[] args, Object target) throws Throwable { System.out.println( "睡覺(jué)后要做美夢(mèng)" ); } } |
一、基于代理的AOP
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
<!-- 創(chuàng)建一個(gè)增強(qiáng) advice --> < bean id = "sleepHelper" class = "com.tgb.springaop.aspect.SleepHelper" /> < bean id = "lina" class = "com.tgb.springaop.service.impl.ChenLliNa" /> <!-- 定義切點(diǎn) 匹配所有的sleep方法--> < bean id = "sleepPointcut" class = "org.springframework.aop.support.JdkRegexpMethodPointcut" > < property name = "pattern" value = ".*sleep" ></ property > </ bean > <!-- 切面 增強(qiáng)+切點(diǎn)結(jié)合 --> < bean id = "sleepHelperAdvisor" class = "org.springframework.aop.support.DefaultPointcutAdvisor" > < property name = "advice" ref = "sleepHelper" /> < property name = "pointcut" ref = "sleepPointcut" /> </ bean > <!-- 定義代理對(duì)象 --> < bean id = "linaProxy" class = "org.springframework.aop.framework.ProxyFactoryBean" > < property name = "target" ref = "lina" /> < property name = "interceptorNames" value = "sleepHelperAdvisor" /> <!-- <property name="proxyInterfaces" value="com.tgb.springaop.service.Sleepable"/> --> </ bean > |
如配置文件中:
pattern屬性指定了正則表達(dá)式,他匹配所有的sleep方法
使用org.springframework.aop.support.DefaultPointcutAdvisor的目的是為了使切點(diǎn)和增強(qiáng)結(jié)合起來(lái)形成一個(gè)完整的切面
最后配置完后通過(guò)org.springframework.aop.framework.ProxyFactoryBean產(chǎn)生一個(gè)最終的代理對(duì)象。
二、純簡(jiǎn)單java對(duì)象切面
純簡(jiǎn)單java對(duì)象切面這話(huà)怎么說(shuō)呢,在我看來(lái)就是相對(duì)于第一種配置,不需要使用代理,,而是通過(guò)spring的內(nèi)部機(jī)制去自動(dòng)掃描,這時(shí)候我們的配置文件就該如下修改:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
<!-- 創(chuàng)建一個(gè)增強(qiáng) advice --> < bean id = "sleepHelper" class = "com.tgb.springaop.aspect.SleepHelper" /> <!-- 目標(biāo)類(lèi) --> < bean id = "lina" class = "com.tgb.springaop.service.impl.ChenLliNa" /> <!-- 配置切點(diǎn)和通知--> < bean id = "sleepAdvisor" class = "org.springframework.aop.support.RegexpMethodPointcutAdvisor" > < property name = "advice" ref = "sleepHelper" ></ property > < property name = "pattern" value = ".*sleep" /> </ bean > <!-- 自動(dòng)代理配置 --> < bean class = "org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator" /> |
是不是相對(duì)于第一種簡(jiǎn)單了許多,不用再去配置代理了。
三、@Aspect注解形式
根據(jù)我們的經(jīng)驗(yàn)也知道,注解的形式相對(duì)于配置文件是簡(jiǎn)單一些的,這時(shí)候需要在已有的方法或類(lèi)上家注解:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
/** * 通過(guò)注解的方式 添加增強(qiáng) */ @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( "睡覺(jué)前要敷面膜" ); } @AfterReturning ( "sleeppoint()" ) public void afterSleep(){ System.out.println( "睡覺(jué)后要做美夢(mèng)" ); } |
配置文件中只需寫(xiě):
1
2
3
4
5
6
7
|
<!--掃描包 --> < context:component-scan base-package = "com.tgb" annotation-config = "true" /> <!-- ASPECTJ注解 --> < aop:aspectj-autoproxy proxy-target-class = "true" /> <!-- 目標(biāo)類(lèi) --> < bean id = "lina" class = "com.tgb.springaop.service.impl.ChenLliNa" /> |
四、注入形式的Aspcet切面
個(gè)人感覺(jué)這個(gè)是最簡(jiǎn)單的也是最常用的,也是最靈活的。配置文件如下:
1
2
3
4
5
6
7
8
9
10
|
<!-- 目標(biāo)類(lèi) --> < bean id = "lina" class = "com.tgb.springaop.service.impl.ChenLliNa" /> < bean id = "sleepHelper" class = "com.tgb.springaop.aspect.SleepHelper02" /> < 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類(lèi)如下:
1
2
3
4
5
6
7
8
9
10
11
12
|
/** * 通過(guò)注解的方式 添加增強(qiáng) */ public class SleepHelper02 { public void beforeSleep(){ System.out.println( "睡覺(jué)前要敷面膜" ); } public void afterSleep(){ System.out.println( "睡覺(jué)后要做美夢(mèng)" ); } } |
是不是看上去都很簡(jiǎn)單呀,這樣是不是大家都會(huì)使用spring aop了?!
關(guān)于如何調(diào)用,這里寫(xiě)了幾個(gè)測(cè)試類(lèi),可以看一下,基本都一樣:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
|
/** * 配置文件 spring_aop.xml 通過(guò)代理 */ @Test public void test(){ ApplicationContext ct = new ClassPathXmlApplicationContext( "spring_aop.xml" ); Sleepable sleeper =(Sleepable) ct.getBean( "linaProxy" ); sleeper.sleep(); } /** * 配置文件 spring_aop_01.xml 簡(jiǎn)答的java對(duì)象 */ @Test public void test01(){ ApplicationContext ct = new ClassPathXmlApplicationContext( "spring_aop_01.xml" ); Sleepable sleeper = (Sleepable)ct.getBean( "lina" ); sleeper.sleep(); } /** * 配置文件 spring_aop_03.xml 通過(guò)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 通過(guò)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(); } |
通過(guò)測(cè)試類(lèi)可以看出,不管以什么樣的方式來(lái)實(shí)現(xiàn)aop他們的使用都是沒(méi)有差別的,這幾個(gè)測(cè)試類(lèi)的結(jié)果都是一樣的: