Spring-AOP

一、概念

Aspect-Oriented-Programming(面向切面编程),一种编程思想。

切面:Aspect,由切入点和额外功能(增强)组成。

作用:解决项目业务中额外功能冗余的问题。

二、业务中存在的问题

public class UserServiceImpl implements UserService {
    private UserDao userDao;

    public void insertUser() {
        System.out.println("调用新建用户服务..."); // 额外功能
        userDao.insertUser(); // 核心功能
    }
}

 

业务中的两大逻辑:核心业务+额外功能,其中额外功能存在大量的代码冗余,使得项目维护存在极大隐患。在OOP(面向对象编程)中,通常会将日志输出等功能再封装一个类来处理冗余问题,今天我们尝试使用spring-aop来解决这个问题。

三、代理

 

在用AOP解决上述提到的问题之前,首先来思考一个问题,什么是代理?业务开发中常见的有静态代理和动态代理,AOP就是使用的动态代理,这里我们将静态代理和动态代理都来分析一下。

1)静态代理

public class UserServiceProxy implements UserService {
    private UserService userService = new UserServiceImpl();

    public void insertUser() {
        System.out.println("调用新建用户服务..."); // 额外功能
        userService.insertUser(); // 核心功能
    }
}

新建一个代理类来处理额外功能,代理类原则上要和目标(核心)保持功能一致,这通常使用接口约束或者继承来实现。代理类虽然从一定程度上简化了目标的业务逻辑,但这仍然没有解决代码冗余的问题。

2)动态代理

public class TestApp {
    @Test
    public void myTest() {
        // 1.目标
        final UserService userService = new UserServiceImpl();
        // 2.额外功能
        InvocationHandler invocationHandler = new InvocationHandler() {
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                System.out.println("调用新建用户服务..."); // 额外功能
                method.invoke(userService, args); // 核心功能
                return null;
            }
        };
        // 3.组装(编织)
        UserService proxy = (UserService) Proxy.newProxyInstance(TestApp.class.getClassLoader(), userService.getClass().getInterfaces(), invocationHandler);
        proxy.insertUser();
    }
}

新建一个测试类,采用jdk中的反射来重新组装一个代理对象,jdk代理通过和目标实现相同的接口来保证功能一致。

三、AOP

 

1.导入依赖

<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-aspects</artifactId>
    <version>5.2.7.RELEASE</version>
</dependency>

2.确认target

<bean id="service" class="service.UserServiceImpl"></bean>

3.实现advice

public class MyBeforeAdvice implements MethodBeforeAdvice {
    public void before(Method method, Object[] objects, Object o) throws Throwable {
        System.out.println("调用新建用户服务..."); // 额外功能
    }
}

同时在applicationContext.xml中声明:

<bean id="before" class="advice.MyBeforeAdvice"></bean>

4.编织weave

<?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">

    <!-- 目标target -->
    <bean id="service" class="service.UserServiceImpl"></bean>

    <!-- 额外功能advice -->
    <bean id="before" class="advice.MyBeforeAdvice"></bean>

    <!-- 编织weave -->
    <aop:config>
        <!-- 切入点:目标中的方法 execution(修饰符 返回值 包.类.方法名(参数表)) -->
        <aop:pointcut id="pc" expression="execution(* service.UserServiceImpl.*(..))"/>
        <!-- 将额外功能编织到切入点中,组装一个新的proxy类 -->
        <aop:advisor advice-ref="before" pointcut-ref="pc"></aop:advisor>
    </aop:config>
</beans>

5.测试

@Test
public void testDynamicProxy() {
    // 启动工厂
    ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
    // 获取代理对象,通过目标id获取
    UserService userService = context.getBean("service", UserService.class);
    userService.insertUser();
    // 关闭工厂
    context.close();
}

 

 

四、补充

 

1)五种额外功能

public class MyAfterAdvice implements AfterReturningAdvice {
    public void afterReturning(Object o, Method method, Object[] objects, Object o1) throws Throwable {
        System.out.println("后置额外功能");
    }
}
// 环绕额外功能
public class MyMethodInterceptor implements MethodInterceptor {
    public Object invoke(MethodInvocation methodInvocation) throws Throwable {
        System.out.println("开始");
        Object ret = methodInvocation.proceed();
        System.out.println("结束");
        return ret;
    }
}

异常额外功能在目标抛出异常时执行,不过大多数的异常处理都是交给controller来做的,了解即可;最终额外功能相当于try,catch,finally中的finally,不管你有没有抛出异常,最终它都执行。

2)三种切入点表达式

execution(修饰符可省略 返回值 *任意 包.类.方法名 *任意(参数表 ..任意))

within描述包和类,类中的所有方法都加入

args描述参数表,符合的方法都加入

联用,不同的表达式之间,可以使用逻辑运算:and or not

 

posted @ 2020-06-27 18:29  viewts  阅读(139)  评论(0编辑  收藏  举报