【Spring】(9.1)AOP - 實現方式一:使用Spring API接口

前言

Spring AOP 的實現有三種方式:使用Spring API 接口、自定義切面、使用註解。這裏將第一種,使用Spring API 接口,另外另種在我的接下來兩篇文章中。

一、使用Spring API接口

使用原生Spring API接口實現數據訪問層前後增加日誌。

注意:使用AOP前,除了spring以外,還需要在maven導入增加一個aspectjweaver的依賴包。

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

項目結構:

在這裏插入圖片描述

UserDao接口:

/** 用戶數據訪問層接口 */
public interface UserDao {
    void ins();
    void del();
    void upd();
    void sel();
}

UserDao接口實現類:

public class UserDaoImpl implements UserDao {
    @Override
    public void ins() {
        System.out.println("insert命令");
    }

    @Override
    public void del() {
        System.out.println("delete命令");
    }

    @Override
    public void upd() {
        System.out.println("update命令");
    }

    @Override
    public void sel() {
        System.out.println("select命令");
    }
}

前置通知:

/** 前置通知(方法前加通知) */
public class LogBefore implements MethodBeforeAdvice {
    /**
     * 參數:method:要執行的目標對象方法。objects:參數(args)。o:目標對象(target)
     */
    @Override
    public void before(Method method, Object[] args, Object target) throws Throwable {
        System.out.println("----- 執行了 " + target.getClass().getName() + " 類的 " + method.getName() + " 方法 -----");
    }
}

後置通知:

/** 後置通知(方法之後加通知) */
public class LogAfter implements AfterReturningAdvice {
    @Override
    public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {
        System.out.println("----- 執行了 " + target.getClass().getName() + " 類的 " + method.getName() + " 方法,返回值類型爲:"+returnValue+" -----");
    }
}

配置文件:

<?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"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd
       http://www.springframework.org/schema/aop
       http://www.springframework.org/schema/aop/spring-aop.xsd">

    <!--註冊bean-->
    <bean id="userDaoImpl" class="com.shengjava.aop.dao.impl.UserDaoImpl"/>
    <bean id="logAfter" class="com.shengjava.aop.log.LogAfter"/>
    <bean id="logBefore" class="com.shengjava.aop.log.LogBefore"/>

    <!--方式一:使用Spring API接口-->
    <!--注意:需要導入aop的約束-->
    <aop:config>
        <!--切入點:expression:表達式,execution(要執行的位置!) *(..):* 表示所有的方法 (..)表示任何參數-->
        <aop:pointcut id="pc-userDao" expression="execution(* com.shengjava.aop.dao.impl.UserDaoImpl.*(..))"/>
        <!--前置通知-->
        <aop:advisor advice-ref="logBefore" pointcut-ref="pc-userDao"/>
        <!--後置通知-->
        <aop:advisor advice-ref="logAfter" pointcut-ref="pc-userDao"/>
    </aop:config>

</beans>

測試:

public class Test {
    public static void main(String[] args) {
        ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
        // 動態代理(代理的是接口,所以類型應該也是接口類型)
        UserDao ud = context.getBean("userDaoImpl", UserDao.class);
        // 執行
        ud.ins();
    }
}

測試結果:使用了Spring的aop,各個類職責非常清晰。

----- 執行了 com.shengjava.aop.dao.impl.UserDaoImpl 類的 ins 方法 -----
insert命令
----- 執行了 com.shengjava.aop.dao.impl.UserDaoImpl 類的 ins 方法,返回值類型爲:null -----

擴展:配置文件中的切入點expression表達式:Combining Pointcut Expressions

相關

我的該分類的其他相關文章,請點擊:【Spring + Spring MVC + MyBatis】文章目錄

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章