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

【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