Spring使用AspectJ注解和XML配置實(shí)現(xiàn)AOP
本文演示的是Spring中使用AspectJ注解和XML配置兩種方式實(shí)現(xiàn)AOP
下面是使用AspectJ注解實(shí)現(xiàn)AOP的Java Project
首先是位于classpath下的applicationContext.xml文件
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd
http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">
<!-- 啟用AspectJ對(duì)Annotation的支持 -->
<aop:aspectj-autoproxy/>
<bean id="userManager" class="com.jadyer.annotation.UserManagerImpl"/>
<bean id="securityHandler" class="com.jadyer.annotation.SecurityHandler"/>
</beans>
然后是服務(wù)層接口以及實(shí)現(xiàn)類(lèi)
package com.jadyer.annotation;
public interface UserManager {
public void addUser(String username, String password);
public void delUser(int userId);
public String findUserById(int userId);
public void modifyUser(int userId, String username, String password);
}
/**
* 上面的UserManager是服務(wù)層的接口
* 下面的UserManagerImpl是服務(wù)層接口的實(shí)現(xiàn)類(lèi)
*/
package com.jadyer.annotation;
public class UserManagerImpl implements UserManager {
public void addUser(String username, String password) {
System.out.println("------UserManagerImpl.addUser() is invoked------");
}
public void delUser(int userId) {
System.out.println("------UserManagerImpl.delUser() is invoked------");
}
public String findUserById(int userId) {
System.out.println("------UserManagerImpl.findUserById() is invoked------");
return "鐵面生";
}
public void modifyUser(int userId, String username, String password) {
System.out.println("------UserManagerImpl.modifyUser() is invoked------");
}
}
接下來(lái)是使用AspectJ注解標(biāo)注的切入類(lèi)
package com.jadyer.annotation;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
@Aspect
public class SecurityHandler {
/**
* 定義Pointcut
* @see Pointcut的名稱(chēng)為addAddMethod(),此方法沒(méi)有返回值和參數(shù)
* @see 該方法就是一個(gè)標(biāo)識(shí),不進(jìn)行調(diào)用
*/
@Pointcut("execution(* add*(..))") //匹配所有以add開(kāi)頭的方法
private void addAddMethod(){};
/**
* 定義Advice
* @see 表示我們的Advice應(yīng)用到哪些Pointcut訂閱的Joinpoint上
*/
//@Before("addAddMethod()")
@After("addAddMethod()")
private void checkSecurity() {
System.out.println("------【checkSecurity is invoked】------");
}
}
最后是客戶端測(cè)試類(lèi)
package com.jadyer.annotation;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
* Spring對(duì)AOP的支持:采用Annotation方式
* @see -------------------------------------------------------------------------------------
* @see Spring提供的AOP功能還是很強(qiáng)大的,支持可配置,它的默認(rèn)實(shí)現(xiàn)使用的就是JDK動(dòng)態(tài)代理
* @see 使用Spring的AOP不需要繼承相關(guān)的東西,也不需要實(shí)現(xiàn)接口
* @see 但有個(gè)前提條件:由于是JDK動(dòng)態(tài)代理,所以若想生成代理,該類(lèi)就必須得實(shí)現(xiàn)一個(gè)接口才行
* @see 如果該類(lèi)沒(méi)有implements接口的話,仍去使用Spring的默認(rèn)AOP實(shí)現(xiàn)時(shí),那么就會(huì)出錯(cuò)
* @see 通常需要生成代理的類(lèi)都是服務(wù)層的類(lèi),所以通常都會(huì)抽一個(gè)接口出來(lái)。即養(yǎng)成面向接口編程的習(xí)慣
* @see -------------------------------------------------------------------------------------
* @see 采用Annotation方式完成AOP示例的基本步驟,如下
* @see 1、Spring2.0的依賴(lài)包配置。新增Annotation支持
* @see * SPRING_HOME//dist//spring.jar
* @see * SPRING_HOME//lib//log4j//log4j-1.2.14.jar
* @see * SPRING_HOME//lib//jakarta-commons//commons-logging.jar
* @see * SPRING_HOME//lib//aspectj//*.jar
* @see 2、將橫切性關(guān)注點(diǎn)模塊化,建立SecurityHandler.java
* @see 3、采用注解指定SecurityHandler為Aspect
* @see 4、采用注解定義Advice和Pointcut
* @see 5、啟用AspectJ對(duì)Annotation的支持,并且將目標(biāo)類(lèi)和Aspect類(lèi)配置到IoC容器中
* @see 6、開(kāi)發(fā)客戶端
* @see -------------------------------------------------------------------------------------
*/
public class Client {
public static void main(String[] args) {
ApplicationContext factory = new ClassPathXmlApplicationContext("applicationContext.xml");
UserManager userManager = (UserManager)factory.getBean("userManager");
userManager.addUser("張起靈", "02200059");
}
}
下面是使用XML配置文件實(shí)現(xiàn)AOP的Java Project
首先是位于src根目錄中的applicationContext-cglib.xml文件
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd
http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">
<!-- 強(qiáng)制使用CGLIB代理 -->
<!-- <aop:aspectj-autoproxy proxy-target-class="true"/> -->
<bean id="userManager" class="com.jadyer.cglib.UserManagerImpl"/>
<bean id="securityHandler" class="com.jadyer.cglib.SecurityHandler"/>
<aop:config>
<aop:aspect id="securityAspect" ref="securityHandler">
<aop:pointcut id="addAddMethod" expression="execution(* add*(..))"/>
<aop:before method="checkSecurity" pointcut-ref="addAddMethod"/>
</aop:aspect>
</aop:config>
</beans>
<!--
匹配add開(kāi)頭的所有的方法
execution(* add*(..))
匹配com.jadyer.servcices.impl包下的所有的類(lèi)的所有的方法
execution(* com.jadyer.servcices.impl.*.*(..))
匹配com.jadyer.servcices.impl包下的add或者del開(kāi)頭的所有的方法
execution(* com.jadyer.servcices.impl.*.add*(..)) || execution(* com.jadyer.servcices.impl.*.del*(..))
-->
然后是服務(wù)層接口以及實(shí)現(xiàn)類(lèi)
package com.jadyer.cglib;
public interface UserManager {
public void addUser(String username, String password);
public void delUser(int userId);
public String findUserById(int userId);
public void modifyUser(int userId, String username, String password);
}
/**
* 上面的UserManager是服務(wù)層接口
* 下面的UserManagerImpl是服務(wù)層接口的實(shí)現(xiàn)類(lèi)
*/
package com.jadyer.cglib;
public class UserManagerImpl {
//implements UserManager {
public void addUser(String username, String password) {
System.out.println("------UserManagerImpl.addUser() is invoked------");
}
public void delUser(int userId) {
System.out.println("------UserManagerImpl.delUser() is invoked------");
}
public String findUserById(int userId) {
System.out.println("------UserManagerImpl.findUserById() is invoked------");
return "張三";
}
public void modifyUser(int userId, String username, String password) {
System.out.println("------UserManagerImpl.modifyUser() is invoked------");
}
}
接著是在applicationContext-cglib.xml中所指定的切入類(lèi)
package com.jadyer.cglib;
import org.aspectj.lang.JoinPoint;
/**
* 將客戶調(diào)用信息傳遞到該Advice中
* @see 可以在Advice中添加一個(gè)JoinPoint參數(shù),取得客戶端調(diào)用的方法名稱(chēng)及參數(shù)值
* @see 以后純粹使用AOP去寫(xiě)類(lèi)似這樣?xùn)|西的情況比較少,我們主要使用Spring提供的事務(wù)
* @see 關(guān)于這個(gè),知道即可。下面是示例代碼
*/
public class SecurityHandler {
private void checkSecurity(JoinPoint joinPoint) {
for (int i=0; i<joinPoint.getArgs().length; i++) {
System.out.println(joinPoint.getArgs()[i]); //獲取客戶端調(diào)用的方法的參數(shù)值
}
System.out.println(joinPoint.getSignature().getName()); //獲取客戶端調(diào)用的方法名稱(chēng)
System.out.println("------【checkSecurity is invoked】------");
}
}
最后是客戶端測(cè)試類(lèi)
package com.jadyer.cglib;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
* @see --------------------------------------------------------------------------------------------------
* @see JDK動(dòng)態(tài)代理和CGLIB代理的差別
* @see 1..JDK動(dòng)態(tài)代理對(duì)實(shí)現(xiàn)了接口的類(lèi)進(jìn)行代理
* @see 2..CGLIB代理可以對(duì)類(lèi)代理,主要對(duì)指定的類(lèi)生成一個(gè)子類(lèi)。由于是繼承,所以目標(biāo)類(lèi)最好不要使用final聲明
* @see --------------------------------------------------------------------------------------------------
* @see 代理方式的選擇
* @see 1..如果目標(biāo)對(duì)象實(shí)現(xiàn)了接口,默認(rèn)情況下會(huì)采用JDK動(dòng)態(tài)代理實(shí)現(xiàn)AOP,亦可強(qiáng)制使用CGLIB生成代理實(shí)現(xiàn)AOP
* @see 2..如果目標(biāo)對(duì)象未實(shí)現(xiàn)接口,那么必須引入CGLIB,這時(shí)Spring會(huì)在JDK動(dòng)態(tài)代理和CGLIB代理之間自動(dòng)切換
* @see 3..比較鼓勵(lì)業(yè)務(wù)對(duì)象是針對(duì)接口編程的,所以鼓勵(lì)使用JDK動(dòng)態(tài)代理。因?yàn)槲覀兯淼哪繕?biāo),一般都是業(yè)務(wù)對(duì)象
* @see --------------------------------------------------------------------------------------------------
* @see 使用CGLIG代理的步驟
* @see 1..新增CGLIB庫(kù):SPRING_HOME//lib//cglib//*.jar
* @see 2..新增配置標(biāo)簽,強(qiáng)制使用CGLIB代理<aop:aspectj-autoproxy proxy-target-class="true"/>
* @see --------------------------------------------------------------------------------------------------
*/
public class Client {
public static void main(String[] args) {
ApplicationContext factory = new ClassPathXmlApplicationContext("applicationContext-cglib.xml");
//當(dāng)UserManagerImpl實(shí)現(xiàn)了UserManager接口的情況下,這時(shí)Spring會(huì)自動(dòng)使用JDK動(dòng)態(tài)代理
//如果項(xiàng)目已經(jīng)引入cglib庫(kù),并在配置文件中強(qiáng)制使用CGLIB代理,此時(shí)Spring才會(huì)使用CGLIB代理
//UserManager userManager = (UserManager)factory.getBean("userManager");
//由于此時(shí)的UserManagerImpl并沒(méi)有實(shí)現(xiàn)UserManager接口,所以接收類(lèi)型就不能再使用UserManager接口
//并且項(xiàng)目中已經(jīng)引入了cglib庫(kù),盡管配置文件中沒(méi)有強(qiáng)制使用CGLIB代理,但Spring會(huì)自動(dòng)使用CGLIB代理
UserManagerImpl userManager = (UserManagerImpl)factory.getBean("userManager");
userManager.addUser("吳三省", "02200059");
}
}
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
java常用工具類(lèi) Random隨機(jī)數(shù)、MD5加密工具類(lèi)
這篇文章主要為大家詳細(xì)介紹了Java常用工具類(lèi),Random隨機(jī)數(shù)工具類(lèi)、MD5加密工具類(lèi),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-05-05
springboot中nacos-client獲取配置的實(shí)現(xiàn)方法
本文主要介紹了springboot中nacos-client獲取配置的實(shí)現(xiàn)方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-04-04
MySQL查詢(xún)字段實(shí)現(xiàn)字符串分割split功能的示例代碼
本文主要介紹了MySQL查詢(xún)字段實(shí)現(xiàn)字符串分割split功能的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01
Spring使用注解更簡(jiǎn)單的讀取和存儲(chǔ)對(duì)象的方法
這篇文章主要介紹了Spring使用注解更簡(jiǎn)單的讀取和存儲(chǔ)對(duì)象的方法,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2023-07-07
SpringBoot實(shí)現(xiàn)掃碼登錄的示例代碼
本文主要介紹了SpringBoot實(shí)現(xiàn)掃碼登錄的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03
通過(guò)第三方接口發(fā)送短信驗(yàn)證碼/短信通知(推薦)
這篇文章主要介紹了通過(guò)第三方接口發(fā)送短信驗(yàn)證碼/短信通知(推薦)的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-08-08
使用Java實(shí)現(xiàn)動(dòng)態(tài)生成MySQL數(shù)據(jù)庫(kù)
這篇文章主要為大家詳細(xì)介紹了如何使用Java實(shí)現(xiàn)動(dòng)態(tài)生成MySQL數(shù)據(jù)庫(kù),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-02-02
IDEA搭建Maven模塊化項(xiàng)目的實(shí)現(xiàn)
本文主要介紹了IDEA搭建Maven模塊化項(xiàng)目的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-05-05
Java變態(tài)跳臺(tái)階實(shí)現(xiàn)思路和代碼
今天小編就為大家分享一篇關(guān)于Java變態(tài)跳臺(tái)階實(shí)現(xiàn)思路和代碼,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2019-01-01

