Spring 09: Spring原生AOP支持

2年前 (2022) 程序员胖胖胖虎阿
183 0 0

常见Spring内置AOP接口

Before通知

  • 在目标方法被调用前调用,
  • 切面需要实现的接口:org.springframework.aop.MethodBeforeAdvice

After通知

  • 在目标方法被调用后调用
  • 切面需要实现的接口:org.springframework.aop.AfterReturningAdvice

Throws通知

  • 在目标方法抛出异常时调用
  • 切面需要实现的接口:org.springframework.aop.ThrowsAdvice

Around通知

  • 环绕通知:拦截对目标对象方法的调用,在被调用方法前后执行切面功能,例如:事务切面就是环绕通知
  • 切面需要实现的接口:org.aopalliance.intercept.MethodInterceptor

原生Before通知示例

  • 其他通知类型同理,不再重复演示

业务接口

package com.example.service;

/**
 * 定义业务接口
 */
public interface Service {
    //购买功能
    default void buy(){}
    //预定功能
    default String order(int orderNums){return null;}
}

业务实现类

package com.example.service.impl;

import com.example.service.Service;

/**
 * 图书业务实现类
 */
public class BookServiceImpl implements Service {
    @Override
    public void buy() {
        System.out.println("图书购买业务....");
    }

    @Override
    public String order(int orderNums) {
        System.out.println("预定图书: " + orderNums + " 册");
        return "预定成功";
    }
}

切面实现类

  • 相当于使用了Spring内置的AOP前置通知接口:org.springframework.aop.MethodBeforeAdvice
package com.example.advice;

import org.springframework.aop.MethodBeforeAdvice;

import java.lang.reflect.Method;
import java.text.SimpleDateFormat;
import java.util.Arrays;
import java.util.Date;

public class LogAdvice implements MethodBeforeAdvice {
    @Override
    public void before(Method method, Object[] args, Object target) throws Throwable {
        //3个参数:目标方法,目标方法返回值,目标对象
        SimpleDateFormat sf = new SimpleDateFormat("yyyy-MM-dd");
        System.out.println("[业务功能名称] :" + method.getName());
        System.out.println("[业务参数信息] :" + Arrays.toString(args));
        System.out.println("[业务办理时间] :" + sf.format(new Date()));
        System.out.println("--------- 具体业务如下 ---------");
    }
}

业务功能和切面功能整合

  • 使用applicationContext.xml,看起来更加直观
    <!--创建业务对象-->
    <bean id="bookServiceTarget" class="com.example.service.impl.BookServiceImpl"/>
    <!--创建切面的对象-->
    <bean id="logAdvice" class="com.example.advice.LogAdvice"/>


    <!-- 相当于创建动态代理对象,用来在底层绑定业务和切面-->
    <bean id="bookService" class="org.springframework.aop.framework.ProxyFactoryBean">
        <!--配置业务接口,底层的jdk动态代理需要用-->
        <property name="interfaces" value="com.example.service.Service"/>

        <!--配置切面,可以有多个-->
        <property name="interceptorNames">
            <list>
                <value>logAdvice</value>
            </list>
        </property>

        <!--待织入切面的业务功能对象,底层的jdk动态代理需要用-->
        <property name="target" ref="bookServiceTarget"/>
    </bean>

对比手写的AOP版本5

将上述applicationContext.xml的内容和AOP版本5中的ProxyFactory对比,上述xml作用就相当于我们手写的ProxyFactory作用:
获取到业务功能对象和切面功能对象,并将他们传给底层来获取动态代理对象,在底层完成切面功能的织入
不管是xml或者通过注解来整合业务和切面,Spring底层都是像手写的AOP版本5一样,通过jdk动态代理来实现的,只不过现在封装起来了

  • AOP版本5中的ProxyFactory代理工厂
package com.example.proxy05;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

/**
 * 代理工厂,获取动态代理对象
 */
public class ProxyFactory {
    //获取jdk动态代理对象
    //Service target 接口类型的业务功能对象
    //Aop aop 接口功能的切面功能对象
    public static Object getProxy(Service target, Aop aop){
        //使用内置类,返回jdk动态代理对象
        return Proxy.newProxyInstance(
                target.getClass().getClassLoader(),
                //获取实现的所有接口
                target.getClass().getInterfaces(),
                //调用目标对象的目标方法
                new InvocationHandler() {
                    @Override
                    public Object invoke(
                            Object obj,
                            Method method,
                            Object[] args) throws Throwable {
                        //存放目标对象的目标方法的返回值
                        Object res = null;
                        try{
                            //切面功能
                            aop.before();
                            //业务功能,根据外部调用的功能,动态代理目标对象被调用的方法
                            res = method.invoke(target, args);
                            //切面功能
                            aop.after();
                        }catch (Exception e){
                            //切面功能
                            aop.exception();
                        }
                        return res;
                    }
                }
        );
    }
}

测试

package test;

import com.example.service.Service;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class TestSpringAOP {
    @Test
    public void testSpringAop(){
        //创建Spring容器
        ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
        //获取动态代理对象
        Service agent = (Service) ac.getBean("bookService");
        //调用业务功能
        String res = agent.order(10);
        System.out.println("结果: " + res);
    }
}

测试输出

  • Spring原生AOP前置通知在业务功能前顺利执行。底层的jdk动态代理也正确的反射调用了外部调用的目标方法,正确接收了参数并给出了返回值
[业务功能名称] :order
[业务参数信息] :[10]
[业务办理时间] :2022-08-23
--------- 具体业务如下 ---------
预定图书: 10 册
结果: 预定成功

Process finished with exit code 0

注意

  • 在开发中一般不常用Spring原生的AOP支持,在需要时,常使用其他专门的AOP框架
  • 手写AOP框架和简单演示Spring内置AOP通知接口是为了更好的理解AOP面向接口编程的思想
版权声明:程序员胖胖胖虎阿 发表于 2022年10月6日 上午9:08。
转载请注明:Spring 09: Spring原生AOP支持 | 胖虎的工具箱-编程导航

相关文章

暂无评论

暂无评论...