欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring中基于XML的面向切面編程(AOP)詳解

 更新時間:2024年04月17日 09:15:26   作者:陳橘又青  
這篇文章主要詳細介紹了Spring中基于XML的面向切面編程(AOP),文中通過代碼示例給大家講解的非常詳細,對大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下

一、基于XML的AOP

1.1、打印日志案例

1.1.1、beans.xml中添加aop的約束

<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        https://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
        https://www.springframework.org/schema/context/spring-context.xsd
        http://www.springframework.org/schema/aop
        https://www.springframework.org/schema/aop/spring-aop.xsd">
</beans>

1.1.2、定義Bean

package cn.bdqn.domain;
public class User {

}
package cn.bdqn.service;
public interface UserService {

    // 保存用戶
    public void save(User user);

    // 根據(jù)id查詢用戶
    public User queryById(Integer id);

    // 查詢?nèi)坑脩?
    public List<User> queryAll();
}
package cn.bdqn.service;
public class UserServiceImpl implements UserService{

    // 保存用戶
    public void save(User user){

    }

    // 根據(jù)id查詢用戶
    public User queryById(Integer id){
        return new User();
    }

    // 查詢?nèi)坑脩?
    public List<User> queryAll(){
        return new ArrayList<User>();
    }
}

1.2、定義記錄日志的類【切面】

package cn.bdqn.advice;

// 定義記錄日志的類,這個類就封裝了我們所有的公共的代碼
public class Logger {

    //  該方法的作用是在切入點方法執(zhí)行之前執(zhí)行
    public void beforePrintLog(){
        System.out.println("開始打印日志啦");
    }
}

1.3、導(dǎo)入AOP的依賴

<dependency>
    <groupId>org.aspectj</groupId>
    <artifactId>aspectjweaver</artifactId>
    <version>1.9.4</version>
</dependency>

1.4、主配置文件中配置AOP

<beans>
  	<!--  1、注冊UserServiceImpl這個Bean  -->
    <bean id="userService" class="cn.bdqn.service.UserServiceImpl"/>

    <!--  2、以下操作都是Spring基于XML的AOP配置步驟
       2.1 把通知/增強Bean也需要注冊到Spring容器中
       2.2 使用<aop:config/>標(biāo)簽來去聲明開始AOP的配置了
       2.3 使用<aop:aspect/>標(biāo)簽來去表示開始配置切面了
        可以想一下:既然要配置切面,那切面就是切入點和通知的結(jié)合,所以肯定需要配置切入點和通知這兩部分
              id屬性:是給切面提供一個唯一標(biāo)識
              ref屬性:是指定通知類bean的Id。
       2.4 在<aop:aspect/>標(biāo)簽的內(nèi)部使用對應(yīng)標(biāo)簽來配置通知的類型
              前置通知/后置通知/異常通知/最終通知
              需求:beforePrintLog方法在切入點方法執(zhí)行之前之前:所以是前置通知
              前置通知:<aop:before/>
                  method屬性:用于指定Logger類中哪個方法是前置通知
                  pointcut屬性:用于指定切入點表達式,該表達式的含義指的是對業(yè)務(wù)層中哪些方法增強

          3、切入點表達式的寫法:
                關(guān)鍵字:execution(表達式)
                表達式:
                    訪問修飾符  方法返回值  包名1.包名2...類名.方法名(參數(shù)列表)
                需求:
                    我現(xiàn)在就想對UserServiceImpl類中的queryAll方法進行攔截
                    public java.util.List cn.bdqn.service.UserServiceImpl.queryAll()
    -->

    <!--  2.1 把通知/增強Bean也需要注冊到Spring容器中  -->
    <bean id="logger" class="cn.bdqn.advice.Logger"/>
    <!--  2.2 使用此標(biāo)簽來去聲明開始AOP的配置了-->
    <aop:config>
        <!--配置切面 -->
        <aop:aspect id="loggerAdvice" ref="logger">
            <!-- 配置通知的類型,并且建立增強方法和切入點方法的關(guān)聯(lián)-->
            <aop:before method="beforePrintLog" 
                        pointcut="execution(public java.util.List cn.bdqn.service.UserServiceImpl.queryAll())"/>
        </aop:aspect>
    </aop:config>
</beans>

1.5、測試

@Test
public void testUserServiceImpl() throws Exception{

   	ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
    UserService userService = (UserService) ac.getBean("userService");

    userService.queryAll();
}

1.6、切入點表達式

問題:我們上面的案例經(jīng)過測試發(fā)現(xiàn)確實在調(diào)用業(yè)務(wù)方法之前增加了日志功能,但是問題是僅僅能針對某一個業(yè)務(wù)方法進行增強,而我們的業(yè)務(wù)方法又有可能有很多,所以顯然一個一個的去配置很麻煩,如何更加靈活的去配置呢?這個就需要使用到切入點表達式

? 語法:execution(表達式)

訪問修飾符  方法返回值  包名1.包名2...類名.方法名(參數(shù)列表)

1.6.1、訪問修飾符可以省略

// 完整寫法
public java.util.List cn.bdqn.service.UserServiceImpl.queryAll())

// 標(biāo)準(zhǔn)寫法
java.util.List cn.bdqn.service.UserServiceImpl.queryAll())

1.6.2、返回值可以使用通配符,表示任意返回值

* cn.bdqn.service.UserServiceImpl.queryAll())

1.6.3、包名可以使用通配符表示任意包。有幾級包,就幾個*

* *.*.*.UserServiceImpl.queryAll())

但是對于包來說,連續(xù)的寫3個*,顯然也是麻煩的,那么可以使用“…”表示當(dāng)前包及其子包。

// 表示的是任意包下的只要有UserServiceImpl類都會對queryAll方法進行增強
* *..UserServiceImpl.queryAll())

1.6.4、類名也可以用*

* *..*.queryAll()

1.6.5、方法也可以用*

* *..*.*()

1.6.6、參數(shù)列表

寫法1、可以直接寫數(shù)據(jù)類型:
             基本類型直接寫名稱           
                  int、double
             引用類型寫包名.類名的方式   
                  java.lang.String、java.util.List
寫法2、可以使用通配符表示任意類型
             前提是必須要有參數(shù)。

寫法3、使用..
             可以使用..表示有無參數(shù)均可,如果有參數(shù)則表示的可以是任意類型    

1.6.7、全通配符寫法

 * *..*.*(..)

1.6.8、使用最多的寫法

實際中的寫法:切到業(yè)務(wù)層實現(xiàn)類下的所有方法。即:

* com.bdqn.service.impl.*.*(..)

1.7、通知類型的使用

1.7.1、在日志類中新增通知方法

// 定義記錄日志的類,這個類就封裝了我們所有的公共的代碼
public class Logger {

    //  該方法的作用是在切入點方法執(zhí)行之前執(zhí)行
    public void beforePrintLog(){
        System.out.println("前置通知(beforePrintLog):開始打印日志啦");
    }

    //  該方法的作用是在切入點方法執(zhí)行之后執(zhí)行
    public void afterReturningPrintLog(){
        System.out.println("后置通知(afterReturningPrintLog):業(yè)務(wù)方法執(zhí)行完了,日志打印");
    }

    //  該方法的作用是在切入點方法執(zhí)行出錯后執(zhí)行
    public void afterThrowingPrintLog(){
        System.out.println("異常通知(afterThrowingPrintLog):業(yè)務(wù)方法出現(xiàn)異常了,日志打印");
    }

    //  該方法的作用是在切入點方法執(zhí)行之后不管有沒有錯誤,都最終要執(zhí)行
    public void afterPrintLog(){
        System.out.println("最終通知(afterPrintLog):業(yè)務(wù)方法不管有沒有異常了,日志打印");
    }
}

1.7.2、配置AOP

<beans>
	<!--  2.1 把通知/增強Bean也需要注冊到Spring容器中  -->
    <bean id="logger" class="cn.bdqn.advice.Logger"/>
    <!--  2.2 使用此標(biāo)簽來去聲明開始AOP的配置了-->
    <aop:config>
        <!--配置切面 -->
        <aop:aspect id="loggerAdvice" ref="logger">
          
            <!-- 配置前置通知:在切入點方法執(zhí)行之前執(zhí)行-->
            <aop:before method="beforePrintLog" 
                        pointcut="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/>
          
            <!-- 后置通知:在切入點方法正常執(zhí)行之后值。它和異常通知永遠只能執(zhí)行一個-->
            <aop:after-returning method="afterReturningPrintLog"  pointcut="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/>
          
            <!--配置異常通知:在切入點方法執(zhí)行產(chǎn)生異常之后執(zhí)行。它和后置通知永遠只能執(zhí)行一個-->
            <aop:after-throwing method="afterThrowingPrintLog"  
                                pointcut="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/>
          
            <!--配置最終通知:無論切入點方法是否正常執(zhí)行它都會在其后面執(zhí)行-->
            <aop:after method="afterPrintLog"
                       pointcut="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/>
            
        </aop:aspect>
    </aop:config>
</beans>

1.7.3、測試

@Test
public void testUserServiceImpl() throws Exception{

        ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
        UserService userService = (UserService) ac.getBean("userService");

        userService.queryAll();
}
/***
	前置通知(beforePrintLog):開始打印日志啦
    查詢?nèi)坑脩魣?zhí)行啦
    后置通知(afterReturningPrintLog):業(yè)務(wù)方法執(zhí)行完了,日志打印
    最終通知(afterPrintLog):業(yè)務(wù)方法不管有沒有異常了,日志打印
**/

1.8、切入點表達式改進

通過11.7可以發(fā)現(xiàn),我們在配置文件中配置了四種通知類型,其中的pointcut配置的是切入點表達式,發(fā)現(xiàn)是一模一樣的,那么有沒有一種改進寫法呢?可以將表達式抽取出來,將來可以引用。

1.8.1、方式一

<beans>
	<!--  1、注冊UserServiceImpl這個Bean  -->
    <bean id="userService" class="cn.bdqn.service.UserServiceImpl"/>

    <!--  2、以下操作都是Spring基于XML的AOP配置步驟-->

    <!--  2.1 把通知/增強Bean也需要注冊到Spring容器中  -->
    <bean id="logger" class="cn.bdqn.advice.Logger"/>
    <!--  2.2 使用此標(biāo)簽來去聲明開始AOP的配置了-->
    <aop:config>
        <!--配置切面 -->
        <aop:aspect id="loggerAdvice" ref="logger">

            <!--
                配置切入點表達式
                    id屬性用于指定切入點表達式的唯一標(biāo)識。
                    expression屬性用于指定表達式內(nèi)容
                此標(biāo)簽寫在aop:aspect標(biāo)簽內(nèi)部只能當(dāng)前切面使用。
           -->
            <aop:pointcut id="loggerPt" 
                          expression="execution(* 																					cn.bdqn.service.UserServiceImpl.queryAll())"/>
            
            <!-- 配置前置通知:在切入點方法執(zhí)行之前執(zhí)行-->
            <aop:before method="beforePrintLog" pointcut-ref="loggerPt"/>
            <!-- 后置通知:在切入點方法正常執(zhí)行之后值。它和異常通知永遠只能執(zhí)行一個-->
            <aop:after-returning method="afterReturningPrintLog" pointcut-ref="loggerPt"/>
            <!--配置異常通知:在切入點方法執(zhí)行產(chǎn)生異常之后執(zhí)行。它和后置通知永遠只能執(zhí)行一個-->
            <aop:after-throwing method="afterThrowingPrintLog" pointcut-ref="loggerPt"/>
            <!--配置最終通知:無論切入點方法是否正常執(zhí)行它都會在其后面執(zhí)行-->
            <aop:after method="afterPrintLog" pointcut-ref="loggerPt"/>
                
        </aop:aspect>
    </aop:config>
</beans>

1.8.2、方式二

對于方式一,我們將aop:pointcut標(biāo)簽寫在了aop:aspect里面,這樣的話這切入點表達式只能被當(dāng)前的切面使用,而如果其他切面想使用就使用不到了,所以我們可以把這個切入點表示再定義到外面。

<beans>
	<bean id="userService" class="cn.bdqn.service.UserServiceImpl"/>

    <!--  2、以下操作都是Spring基于XML的AOP配置步驟-->

    <!--  2.1 把通知/增強Bean也需要注冊到Spring容器中  -->
    <bean id="logger" class="cn.bdqn.advice.Logger"/>
    <!--  2.2 使用此標(biāo)簽來去聲明開始AOP的配置了-->
    <aop:config>

        <!--
                配置切入點表達式
                    id屬性用于指定切入點表達式的唯一標(biāo)識。
                    expression屬性用于指定表達式內(nèi)容
                此標(biāo)簽寫在aop:aspect標(biāo)簽外面,那么所有的切面都可以使用。
          -->
        <aop:pointcut id="loggerPt" 
                      expression="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/>

        <!--配置切面 -->
        <aop:aspect id="loggerAdvice" ref="logger">

            <!-- 配置前置通知:在切入點方法執(zhí)行之前執(zhí)行-->
            <aop:before method="beforePrintLog" pointcut-ref="loggerPt"/>
            <!-- 后置通知:在切入點方法正常執(zhí)行之后值。它和異常通知永遠只能執(zhí)行一個-->
            <aop:after-returning method="afterReturningPrintLog" pointcut-ref="loggerPt"/>
            <!--配置異常通知:在切入點方法執(zhí)行產(chǎn)生異常之后執(zhí)行。它和后置通知永遠只能執(zhí)行一個-->
            <aop:after-throwing method="afterThrowingPrintLog" pointcut-ref="loggerPt"/>
            <!--配置最終通知:無論切入點方法是否正常執(zhí)行它都會在其后面執(zhí)行-->
            <aop:after method="afterPrintLog" pointcut-ref="loggerPt"/>
                
        </aop:aspect>
    </aop:config>
</beans>

1.9、環(huán)繞通知

1.9.1、在日志記錄類中新增環(huán)繞通知

public class Logger {
    // 環(huán)繞通知
    public void aroundPrintLog(){
        System.out.println("環(huán)繞通知....aroundPrintLog.....");
    }
}

1.9.2、AOP配置環(huán)繞通知

<beans>
   <!--  1、注冊UserServiceImpl這個Bean  -->
    <bean id="userService" class="cn.bdqn.service.UserServiceImpl"/>

    <!--  2、以下操作都是Spring基于XML的AOP配置步驟-->
    <!--  2.1 把通知/增強Bean也需要注冊到Spring容器中  -->
    <bean id="logger" class="cn.bdqn.advice.Logger"/>
    <!--  2.2 使用此標(biāo)簽來去聲明開始AOP的配置了-->
    <aop:config>

        <!--    配置切入點表達式    -->
        <aop:pointcut id="loggerPt" 
                      expression="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/>

        <!--配置切面 -->
        <aop:aspect id="loggerAdvice" ref="logger">

            <!-- 環(huán)繞通知-->
            <aop:around method="aroundPrintLog" pointcut-ref="loggerPt"/>
                
        </aop:aspect>
    </aop:config>
</beans>

1.9.3、測試1

@Test
public void testUserServiceImpl() throws Exception{

        ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
        UserService userService = (UserService) ac.getBean("userService");

        userService.queryAll();
}
/**
	環(huán)繞通知....aroundPrintLog.....
	發(fā)現(xiàn):僅僅打印了環(huán)繞通知的代碼。當(dāng)我們配置了環(huán)繞通知之后,切入點方法沒有執(zhí)行,而通知方法執(zhí)行了
*/

1.9.4、解決

Spring框架為我們提供了一個接口:ProceedingJoinPoint。該接口有一個方法proceed(),此方法就相當(dāng)于明確調(diào)用切入點方法。該接口可以作為環(huán)繞通知的方法參數(shù),在程序執(zhí)行時,spring框架會為我們提供該接口的實現(xiàn)類供我們使用。

public class Logger {
    // 環(huán)繞通知
    public Object aroundPrintLog(ProceedingJoinPoint pjp){
        Object result = null;
        try{
            Object[] args = pjp.getArgs();
            System.out.println(pjp.getSignature().getName());
            System.out.println("前置");
            result = pjp.proceed(args);
            System.out.println("后置");
            return result;
        }catch (Throwable t){
            System.out.println("異常");
            throw new RuntimeException(t);
        }finally {
            System.out.println("最終");
        }
    }
}
/**
	環(huán)繞通知:它是spring框架為我們提供的一種可以在代碼中手動控制增強方法何時執(zhí)行的方式。
*/

以上就是Spring中基于XML的面向切面編程(AOP)詳解的詳細內(nèi)容,更多關(guān)于Spring 于XML面向切面編程的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • SpringBoot與Dubbo整合的方式詳解

    SpringBoot與Dubbo整合的方式詳解

    這篇文章主要介紹了SpringBoot與Dubbo整合的方式詳解,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2019-09-09
  • 從log4j切換到logback后項目無法啟動的問題及解決方法

    從log4j切換到logback后項目無法啟動的問題及解決方法

    這篇文章主要介紹了從log4j切換到logback后項目無法啟動的問題及解決方法,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2023-01-01
  • Java JDBC連接Kerberos認證的HIVE和Impala方式

    Java JDBC連接Kerberos認證的HIVE和Impala方式

    本文主要介紹了HiveJDBC和ImpalaJDBC的使用方法,包括版本對應(yīng)、Maven安裝、主機名配置、端口開通、JDBC連接和Kerberos認證等
    2025-02-02
  • Java數(shù)組去重復(fù)的18種方法示例

    Java數(shù)組去重復(fù)的18種方法示例

    這篇文章主要為大家介紹了Java數(shù)組去重復(fù)的18種寫法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-08-08
  • Java爬取豆瓣電影數(shù)據(jù)的方法詳解

    Java爬取豆瓣電影數(shù)據(jù)的方法詳解

    這篇文章主要介紹了Java爬取豆瓣電影數(shù)據(jù)的方法,結(jié)合實例形式詳細分析了Java爬取豆瓣電影數(shù)據(jù)相關(guān)原理、操作步驟、實現(xiàn)技巧與注意事項,需要的朋友可以參考下
    2020-04-04
  • Java中使用DOM和SAX解析XML文件的方法示例

    Java中使用DOM和SAX解析XML文件的方法示例

    這篇文章主要介紹了Java中使用DOM和SAX解析XML文件的方法示例,通過實例文章中最后也給出了一些對比結(jié)論,需要的朋友可以參考下
    2015-11-11
  • Java基本語法筆記(菜鳥必看篇)

    Java基本語法筆記(菜鳥必看篇)

    下面小編就為大家?guī)硪黄狫ava基本語法筆記(菜鳥必看篇)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-05-05
  • Java多線程+鎖機制實現(xiàn)簡單模擬搶票的項目實踐

    Java多線程+鎖機制實現(xiàn)簡單模擬搶票的項目實踐

    鎖是一種同步機制,用于控制對共享資源的訪問,在線程獲取到鎖對象后,可以執(zhí)行搶票操作,本文主要介紹了Java多線程+鎖機制實現(xiàn)簡單模擬搶票的項目實踐,具有一定的參考價值,感興趣的可以了解一下
    2024-02-02
  • 使用HttpSessionListener監(jiān)聽器實戰(zhàn)

    使用HttpSessionListener監(jiān)聽器實戰(zhàn)

    這篇文章主要介紹了使用HttpSessionListener監(jiān)聽器實戰(zhàn),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-03-03
  • 詳解 Java繼承關(guān)系下的構(gòu)造方法調(diào)用

    詳解 Java繼承關(guān)系下的構(gòu)造方法調(diào)用

    這篇文章主要介紹了詳解 Java繼承關(guān)系下的構(gòu)造方法調(diào)用的相關(guān)資料,希望通過本文能幫助到大家,讓大家理解掌握這部分內(nèi)容,需要的朋友可以參考下
    2017-10-10

最新評論