實(shí)例講解Java的Spring框架中的AOP實(shí)現(xiàn)
簡(jiǎn)介
面向切面編程(AOP)提供另外一種角度來(lái)思考程序結(jié)構(gòu),通過(guò)這種方式彌補(bǔ)了面向?qū)ο缶幊蹋∣OP)的不足。 除了類(classes)以外,AOP提供了 切面。切面對(duì)關(guān)注點(diǎn)進(jìn)行模塊化,例如橫切多個(gè)類型和對(duì)象的事務(wù)管理。 (這些關(guān)注點(diǎn)術(shù)語(yǔ)通常稱作 橫切(crosscutting) 關(guān)注點(diǎn)。)
Spring的一個(gè)關(guān)鍵的組件就是 AOP框架。 盡管如此,Spring IoC容器并不依賴于AOP,這意味著你可以自由選擇是否使用AOP,AOP提供強(qiáng)大的中間件解決方案,這使得Spring IoC容器更加完善。
Spring 2.0 AOP:
Spring 2.0 引入了一種更加簡(jiǎn)單并且更強(qiáng)大的方式來(lái)自定義切面,用戶可以選擇使用基于模式(schema-based)的方式或者使用@AspectJ注解。 對(duì)于新的應(yīng)用程序,如果用戶使用Java 5開(kāi)發(fā),我們推薦用戶使用@AspectJ風(fēng)格,否則可以使用基于模式的風(fēng)格。 這兩種風(fēng)格都完全支持通知(Advice)類型和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)之上。
允許用戶實(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è)基本的類。
接口類:
/**
* 定義一個(gè)接口
*/
public interface Sleepable {
/**
* 睡覺(jué)方法
*/
void sleep();
}
實(shí)現(xiàn)類:
/**
* 本人實(shí)現(xiàn)睡覺(jué)接口
*/
public class ChenLliNa implements Sleepable {
@Override
public void sleep() {
// TODO Auto-generated method stub
System.out.println("乖,該睡覺(jué)了!");
}
}
增強(qiáng)類:
/**
* 定義一個(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
<!-- 創(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ì)象切面這話怎么說(shuō)呢,在我看來(lái)就是相對(duì)于第一種配置,不需要使用代理,,而是通過(guò)spring的內(nèi)部機(jī)制去自動(dòng)掃描,這時(shí)候我們的配置文件就該如下修改:
<!-- 創(chuàng)建一個(gè)增強(qiáng) advice --> <bean id ="sleepHelper" class="com.tgb.springaop.aspect.SleepHelper"/> <!-- 目標(biāo)類 --> <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í)候需要在已有的方法或類上家注解:
/**
* 通過(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ě):
<!--掃描包 -->
<context:component-scan base-package="com.tgb" annotation-config="true"/>
<!-- ASPECTJ注解 -->
<aop:aspectj-autoproxy proxy-target-class="true" />
<!-- 目標(biāo)類 -->
<bean id="lina" class="com.tgb.springaop.service.impl.ChenLliNa"/>
四、注入形式的Aspcet切面
個(gè)人感覺(jué)這個(gè)是最簡(jiǎn)單的也是最常用的,也是最靈活的。配置文件如下:
<!-- 目標(biāo)類 -->
<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類如下:
/**
* 通過(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è)試類,可以看一下,基本都一樣:
/**
* 配置文件 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)實(shí)現(xiàn)aop他們的使用都是沒(méi)有差別的,這幾個(gè)測(cè)試類的結(jié)果都是一樣的:

相關(guān)文章
數(shù)據(jù)結(jié)構(gòu)與算法之手撕排序算法
排序算法看似簡(jiǎn)單,其實(shí)不同的算法中蘊(yùn)涵著經(jīng)典的算法策略。通過(guò)熟練掌握排序算法,就可以掌握基本的算法設(shè)計(jì)思想,本文主要介紹了Java中的排序算法,需要的朋友歡迎閱讀2023-04-04
Java以編程方式實(shí)現(xiàn)JAR文件的創(chuàng)建
在這篇文章中,我們將為大家詳細(xì)介紹一下利用Java語(yǔ)言以編程方式創(chuàng)建jar文件的過(guò)程。文中的示例代碼講解詳細(xì),感興趣的可以了解一下2022-07-07
Java中關(guān)于http請(qǐng)求獲取FlexManager某設(shè)備分組監(jiān)控點(diǎn)
這篇文章主要介紹了Java中關(guān)于http請(qǐng)求獲取FlexManager某設(shè)備分組監(jiān)控點(diǎn),本文僅僅介紹了使用http請(qǐng)求獲取FlexManager平臺(tái)某個(gè)FBox盒子即某設(shè)備的監(jiān)控點(diǎn)分組的分組下的所有監(jiān)控點(diǎn)信息,需要的朋友可以參考下2022-10-10
SpringBoot2.3整合redis緩存自定義序列化的實(shí)現(xiàn)
這篇文章主要介紹了SpringBoot2.3整合redis緩存自定義序列化的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08
IntelliJ IDEA 統(tǒng)一設(shè)置編碼為utf-8編碼的實(shí)現(xiàn)
這篇文章主要介紹了IntelliJ IDEA 統(tǒng)一設(shè)置編碼為utf-8編碼的實(shí)現(xiàn),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2020-06-06
Java實(shí)現(xiàn)藍(lán)橋杯G將軍的示例代碼
這篇文章主要介紹了Java實(shí)現(xiàn)藍(lán)橋杯G將軍的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-02-02

