spring AOP

AOP简介

AOP(面向切面编程)是一种编程范式,它可以有效地解耦业务逻辑和系统框架,将通用的功能代码(如日志记录、安全控制、事务管理等)从业务代码中剥离出来,以模块化的方式实现代码的复用和扩展。

AOP通过将横切关注点(cross-cutting concerns)与主要业务逻辑分离,使得系统更加灵活、可维护、可测试和可扩展。在AOP中,横切关注点是指那些不是直接与业务逻辑相关的代码,它们可能会出现在多个地方,造成代码重复,同时也难以维护和更新。通过将这些代码抽离出来,可以使得系统更加清晰、简洁、易于理解。

AOP可以通过不同的实现方式来实现横切关注点的剥离,如基于代理(Proxy)的方式、基于字节码操作的方式等。常见的AOP框架包括Spring AOP、AspectJ等。

代理模式

概念

二十三种设计模式中的一种,属于结构型模式。它的作用就是通过提供一个代理类,让我们在调用目标方法的时候,不再是直接对目标方法进行调用,而是通过代理类间接调用。让不属于目标方法核心逻辑的代码从目标方法中剥离出来一一解耦。调用目标方法时先调用代理对象的方法,减少对目标方法的调用和打扰,同时让附加功能能够集中在一起也有利于统一维护。

请添加图片描述
请添加图片描述

静态代理

静态代理确实实现了解耦,但是由于代码都写死了,完全不具备任何的灵活性。就拿日志功能来说,将来其他地方也需要附加日志,那还得再声明更多个静态代理类,那就产生了大量重复的代码,日志功能还是分散的,没有统一管理。
提出进一步的需求:将日志功能集中到一个代理类中,将来有任何日志需求,都通过这一个代理类来实现。这就需要使用动态代理技术了

public class CalculatorStaticProxy implements Calculator{
    //被代理目标对象传递过来
    private Calculator calculator;

    public CalculatorStaticProxy(Calculator calculator) {
        this.calculator = calculator;
    }

    @Override
    public int add(int i, int j) {
        //非核心内容
        System.out.println("[日志] add 方法开始了,参数是:"+ i+","+ j);
        //核心业务:调用目标方法
        int add = calculator.add(i, j);
        //非核心内容
        System.out.println("[日志] add 方法结束了,结果是:" + add);
        return 0;
    }

动态代理

在这里插入图片描述

package com.xinzhi.aop.example;

public interface Calculator {
    int add(int i, int j);

    int sub(int i, int j);

    int mul(int i, int j);

    int div(int i, int j);
}
package com.xinzhi.aop.example;
//基本实现类
public class CalculatorImpl  implements Calculator {
    @Override
    public int add(int i, int j) {
        int result = i+j;
        System.out.println("方法内部;result="+result);
        return result;
    }

    @Override
    public int sub(int i, int j) {
        int result = i-j;
        System.out.println("方法内部;result="+result);
        return result;
    }

    @Override
    public int mul(int i, int j) {
        int result = i*j;
        System.out.println("方法内部;result="+result);
        return result;
    }

    @Override
    public int div(int i, int j) {
        int result = i/j;
        System.out.println("方法内部;result="+result);
        return result;
    }
}

package com.xinzhi.aop.example;

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

public class ProxyFactory {
    //目标对象
    private Object target;

    public ProxyFactory(Object target) {
        this.target = target;
    }

    //返回代理对象
    public Object getProxy() {
        //Proxy.newProxyInstance()
        //需要三个参数
        //1 ClassLoader:加载动态生成代理类的加载器
        //2 Class[] interfaces:目标对象实现的所有接口的class类型数组
        //3 InvocationHandLer:设置代理对象实现目标对象方法的过程

        //1 ClassLoader:加载动态生成代理类的加载器
        ClassLoader classLoader = target.getClass().getClassLoader();
        //2 Class[] interfaces:目标对象实现的所有接口的class类型数组
        Class<?>[] interfaces = target.getClass().getInterfaces();
        //3 InvocationHandLer:设置代理对象实现目标对象方法的过程
        InvocationHandler invocationHandler = new InvocationHandler() {
            //1 Object proxy代理对象
            //2 Method method需要重写目标对象的方法
            //3 Object[] args:method方法里的参数
            @Override
            public Object invoke(Object proxy,
                                 Method method,
                                 Object[] args) throws Throwable {


                //方法调用前输出
                System.out.println("[动态代理][日志]"+method.getName()+",参数:"+Arrays.toString(args));
                //调用目标方法
                Object result = method.invoke(target, args);
                //方法调用后输出
                System.out.println("[动态代理][日志]"+method.getName()+",结果:"+result);
                return result;
            }
        };
        return Proxy.newProxyInstance(classLoader,interfaces,invocationHandler);
    }
}

package com.xinzhi.aop.example;

public class TestCal {
    public static void main(String[] args) {
        ProxyFactory proxyFactory = new ProxyFactory(new CalculatorImpl());
        Calculator proxy = (Calculator) proxyFactory.getProxy();
        proxy.add(1,2);
    }
}

AOP 相关概念

Target(目标对象):代理的目标对象 98K

Proxy (代理):一个类被 AOP 织入增强后,就产生一个结果代理类 安装了8倍镜的98K

Joinpoint(连接点):所谓连接点是指那些被拦截到的点。在spring中,这些点指的是方法,因为spring只支持方法类型的连接点。 可以安装8倍境的地方

Pointcut(切入点):所谓切入点是指我们要对哪些 Joinpoint 进行拦截的定义。 安装了8被镜的地方

Advice(通知/ 增强):所谓通知是指拦截到 Joinpoint 之后所要做的事情就是通知 8倍镜

Aspect(切面):是切入点和通知的结合

Weaving(织入):是指把增强应用到目标对象来创建新的代理对象的过程。spring采用动态代理织入,而AspectJ采用编译期织入和类装载期织入。 安装的过程

xml的AOP

1 书写步骤

  • 1 导入 AOP 相关坐标
  <!--导入spring的context坐标,context依赖aop-->
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-context</artifactId>
      <version>5.0.5.RELEASE</version>
    </dependency>
    <!-- aspectj的织入 -->
    <dependency>
      <groupId>org.aspectj</groupId>
      <artifactId>aspectjweaver</artifactId>
      <version>1.8.13</version>
    </dependency>

  • 2 创建目标接口和目标类(内部有切点)
/**
 * 接口类
 */
public interface TargetInterface {

    void say();
}

/**
 * 目标类
 */
public class Target implements TargetInterface {
 	@Override
    public void say() {
        try {
            Thread.sleep(2000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("目标类执行了");
    }
}

  • 3 创建切面类(内部有增强方法)
public class MyAspect {
     //前置增强方法
    public void before(){
        System.out.println("前置代码增强了");
    }
    //后置增强方法
    public void after(){
        System.out.println("后置");
    }
    //环绕增强
    public Object around(ProceedingJoinPoint p){
        long start = System.currentTimeMillis();
        String className = p.getSignature().getDeclaringTypeName();
        String method =  p.getSignature().getName();
        Object proceed = null;
        try {
            proceed = p.proceed();
        } catch (Throwable throwable) {
            throwable.printStackTrace();
        }
        long end = System.currentTimeMillis();
        System.out.println("类名:" + className);
        System.out.println("方法名是:" + method);
        System.out.println("执行时间长是:"+(end-start));
        return proceed;
    }
}
  • 4 将目标类和切面类的对象创建权交给 spring
<!--配置目标类-->
<bean id="target" class="com.xinzih.service.impl.Target"/>
<!--配置切面类-->
<bean id="myAspect" class="com.xinzih.aspect.MyAspect"/>
  • 5 在 applicationContext.xml 中配置织入关系
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       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 id="myAspect" class="com.xinzhi.aspect.MyAspect" />

    <bean id="target" class="com.xinzhi.target.impl.TargetInterfaceImpl" />

    <aop:config>
        <aop:aspect ref="myAspect">
<!--            <aop:before method="before" pointcut="execution(* com.xinzhi.target.impl.*.*(..))"/>-->
<!--            <aop:after method="after" pointcut="execution(* com.xinzhi.target.impl.*.*(..))"/>-->
            <aop:around method="around" pointcut="execution(* com.xinzhi.target.impl.*.*(..))"/>
        </aop:aspect>
    </aop:config>

</beans>
  • 6 测试代码
public static void main( String[] args ) {
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("application.xml");
        TargetInterface bean = (TargetInterface)applicationContext.getBean("target");
        bean.say();
 }

2 切点表达式

execution([修饰符] 返回值类型 包名.类名.方法名(参数))
	常用的书写格式:
	execution(* 包名.*.*(..))
  • 访问修饰符可以省略
  • 返回值类型、包名、类名、方法名可以使用星号* 代表任意
  • 包名与类名之间一个点 . 代表当前包下的类,两个点 … 表示当前包及其子包下的类
  • 参数列表可以使用两个点 … 表示任意个数,任意类型的参数列表

三 注解的AOP

1 书写步骤

  • 1创建目标接口和目标类(内部有切点)

  • 2 创建切面类(内部有增强方法)

  • 3 将目标类和切面类的对象创建权交给 spring

  • 4 在切面类中使用注解配置织入关系

  • 5 在配置文件中开启组件扫描和 AOP 的自动代理

  • 6 测试

/**
 * 目标类
 */
@Component("target")
public class Target implements TargetInterface {

    public void say() {
        System.out.println("目标类执行了");
    }
}

@Component
@Aspect
public class MyAspect {
    //前置增强方法
    @Before("execution(* com.xinzhi.target.impl.*.*(..))")
    public void before(){
        System.out.println("前置代码增强了");
    }
    //后置增强方法
    @After("execution(* com.xinzhi.target.impl.*.*(..))")
    public void after(){
        System.out.println("后置");
    }
    //环绕增强
    @Around("execution(* com.xinzhi.target.impl.*.*(..))")
    public Object around(ProceedingJoinPoint p){
        long start = System.currentTimeMillis();
        String className = p.getSignature().getDeclaringTypeName();
        String method =  p.getSignature().getName();
        Object proceed = null;
        try {
            proceed = p.proceed();
        } catch (Throwable throwable) {
            throwable.printStackTrace();
        }
        long end = System.currentTimeMillis();
        System.out.println("类名:" + className);
        System.out.println("方法名是:" + method);
        System.out.println("执行时间长是:"+(end-start));
        return proceed;
    }
}

<!--组件扫描-->
<context:component-scan base-package="com.xinzhi"/>
<!--aop的自动代理-->
<aop:aspectj-autoproxy></aop:aspectj-autoproxy>
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值