SpringBoot | 1.3 约定编程Spring AOP

简介: 前面聊过Spring的一个很重要的概念,IoC控制反转,接下来就是AOP了;注:在说明注解时,第一点加粗为注解中文含义,第二点为一般加在哪身上,缩进或代码块为示例,如:**@注解** - **中文含义** - 加在哪 - 其他…… - `语句示例` ```java //代码示例 ```

1. AOP切面编程

面向切面编程,是利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。

可以理解成“增强方法”,即:不通过修改源代码方式,在主干功能里面添加新功能。

Spring AOP是一种约定流程的编程,我们将这种流程成为约定,如下流程所示:


20210614001813921.png

为什么使用AOP?
AOP最典型的应用实际就是数据库的管控,用户信息与用户角色信息一般要求要么一起成功,要么一起失败。这是OOP(面向对象编程)不能实现的。

除此之外,AOP可以减少大量重复的工作。

@Transactional

  • 事务的
  • 常用于需要事务处理的方法上;
  • 大致流程为:Spring将方法织入以下流程图中,默认实现数据库连接的打开与关闭以及事务,也就是说将大量重复的try-catch-finally流程通过约定的方式抽取,给予默认实现;
  • 这样我们只需要专注SQL语句的编写即可。


20210614163838914.png


2. JDK动态代理底层原理

AOP底层使用动态代理,实际上有两种情况动态代理;第一种是有接口情况,使用JDK动态代理,创建接口实现类代理对象,增强类的方法;第二种没有接口情况,使用CGLIB动态代理,创建子类的代理对象,增强类的方法。这里只讨论有接口的JDK动态代理方式。

在JDK动态代理场景中,主要使用java.lang.reflect.Proxy类里面的newProxyInstance方法创建代理对象

//Proxy类里newProxyInstance方法的源码
@CallerSensitive
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) {
    Objects.requireNonNull(h);
    Class<?> caller = System.getSecurityManager() == null ? null : Reflection.getCallerClass();
    Constructor<?> cons = getProxyConstructor(caller, loader, interfaces);
    return newProxyInstance(caller, cons, h);
}

newProxyInstance方法的作用是:返回指定接口的代理类的实例,该接口方法调用分派给指定的调用处理程序。

查阅源码可知,在该方法中,有三个参数:

  • 第一参数:类加载器;
  • 第二参数:增强方法所在的类,这个类实现的接口,支持多个接口;
  • 第三参数:实现接口InvocationHandler,创建代理对象,写增强的部分;
//InvocationHandler接口源码:
public interface InvocationHandler {
    Object invoke(Object var1, Method var2, Object[] var3) throws Throwable;
}

查阅源码可知,该接口定义了一个invoke方法,这个方法就是实现代理对象的逻辑,也是我们要实现的方法

然后通过目标对象(target)、方法(method)和参数(args)反射方法运行。


3. 使用AOP开发

在本小节里比较重要的是第1)和2)点,第3)点不常用。

1) 确定连接点

因为Spring AOP只能对方法进行拦截,因此需要确定拦截什么方法,让其编入约定的流程中。

这里只专注@AspectJ的注解方式,对userService.printUser()方法进行增强。

用户服务接口

public interface UserService {
    public void printUser(User user);
}

用户服务接口实现类

@Service
public class UserServiceImpl implements UserService{
    @Override
    public void printUser(User user) { 
        if( user == null){
            throw new RuntimeException("检查用户参数是否为空");
        }
        System.out.print("id = " + user.getId());
        System.out.print("\tusername = " + user.getUseranme());
        System.out.print("\tnote = " + user.getNote());
    }
}

2) 配置不同类型的通知(切点、切面、环绕等通知)

通过切面可以描述AOP其他信息,用以描述流程的织入。

可以理解成在这里在这里实现对Bean的功能增强。

@Aspect

  • 切面
  • 常用于增强类上;
  • Spring以@Aspect注解声明切面;

@Order

  • 增强次序
  • 常用于增强类上;
  • 当多个增强类对同一个方法进行增强,设置增强类优先级,数字越小优先级越高;
  • 也可以使用Ordered接口实现(不推荐,推荐注解)

    @Aspect
    //@Order(1) //该注解相当于下面对接口的实现
    public class MyAspect implements Ordered {
        //指定顺序
        @Override
        public int getOrdered() {
            return 1;
        }
    }

@Pointcut

  • 切点
  • 用于方法上;
  • 描述哪些类的哪些方法需要启用AOP编程,在后面通知注解中可以使用方法名称pointCut()定义;
  • 本例中前置、后置、最终和异常通知使用了切点描述,环绕通知类型也可转成切点描述;

     @Pointcut("execution(* com.dlhjw.springboot.service.impl.UserServiceImpl.printUser(..))")
     public void pointCut() {
     }

@Before、@AfterReturning、@After、@AfterThrowing、@Around

通知类型:均作用在方法上
注解 说明
@Before 前置通知
@AfterReturning 后置通知
@After 最终通知
@AfterThrowing 异常通知
@Around 环绕通知,需要大幅修改原有目标对象时使用

对于非环绕通知,可以使用JoinPoint连接点获取需要增强方法的参数:

  • 例子在下方前置通知上

对于环绕通知,可以使用ProceedingsJoinPoint类型的参数。

@Component
@Aspect
//@Order(1)
public class MyAspect{
    //相同切入点抽取,并命名为pointCut()
    @Pointcut("execution(* com.dlhjw.springboot.service.impl.UserServiceImpl.printUser(..))")
    public void pointCut() {
    }

    //前置通知
    @Before("pointCut()" && args(user))
    public void beforeParam(JoinPoint point, User user) {
        //获取到args[0] = user
        Object[] args = point.getArgs();
        System.out.println("before.........");
    }

    //后置通知(返回通知)
    @AfterReturning("pointCut()")
    public void afterReturning() {
        System.out.println("afterReturning.........");
    }

    //最终通知
    @After("pointCut()")
    public void after() {
        System.out.println("after.........");
    }

    //异常通知
    @AfterThrowing("pointCut()")
    public void afterThrowing() {
        System.out.println("afterThrowing.........");
    }

    //环绕通知
    @Around("execution(* com.dlhjw.springboot.service.impl.UserServiceImpl.printUser(..))")
    public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
        System.out.println("环绕之前.........");
        //被增强的方法执行
        proceedingJoinPoint.proceed();
        System.out.println("环绕之后.........");
    }
}

其中execution(* com.dlhjw.springboot.service.impl.UserServiceImpl.printUser(..))表示正则式:

  • execution 表示在执行时拦截正则匹配的方法;
  • * 表示任意返回类型的方法;
  • com.dlhjw.springboot.service.impl.UserServiceImpl 表示指定目标对象的全类名;
  • printUser 表示指定目标对象的方法;
  • (..) 表示任意参数进行匹配。

对于正则式而言,还可以使用AspectJ的指示器:


2021061419402844.png


*3) 引入新的类增强服务

当我们需要利用外部类对方法进行增强时,即:对userService.printUser()方法进行非空校验时。

用户检测的接口UserValidator

public interface UserValidator {
    //检测用户对象是否为空
    public boolean validate(User user);
}

UserValidator的实现类

public class UserValidatorImpl implements UserValidator {
    @Override
    public boolean validate(User user) {
        System.out.println("引入新的接口:" + UserValidator.class.getSimpleName());
        return user != null;
    }
}

@DeclareParents

  • 引入声明
  • 用于属性上;
  • 引入新的类来增强服务,其必须有两个配置属性value和defaultImpl;

    • value 指要增强功能的目标对象;
    • defaultImpl 引入增强功能的类,这里配置UserValidatorImpl,用来提供校验用户是否为空的功能。

      @Component
      @Aspect
      public class MyAspect{
          @DeclareParents(value= "com.dlhjw.springboot.service.impl.UserServiceImpl+", defaultImpl=UserValidatorImpl.class)
          public UserValidator userValidator;
          
            ......
      }
相关文章
|
28天前
|
XML Java 数据安全/隐私保护
Spring Aop该如何使用
本文介绍了AOP(面向切面编程)的基本概念和术语,并通过具体业务场景演示了如何在Spring框架中使用Spring AOP。文章详细解释了切面、连接点、通知、切点等关键术语,并提供了完整的示例代码,帮助读者轻松理解和应用Spring AOP。
Spring Aop该如何使用
|
1月前
|
安全 Java 编译器
什么是AOP面向切面编程?怎么简单理解?
本文介绍了面向切面编程(AOP)的基本概念和原理,解释了如何通过分离横切关注点(如日志、事务管理等)来增强代码的模块化和可维护性。AOP的核心概念包括切面、连接点、切入点、通知和织入。文章还提供了一个使用Spring AOP的简单示例,展示了如何定义和应用切面。
91 1
什么是AOP面向切面编程?怎么简单理解?
|
2月前
|
存储 缓存 Java
Spring高手之路23——AOP触发机制与代理逻辑的执行
本篇文章深入解析了Spring AOP代理的触发机制和执行流程,从源码角度详细讲解了Bean如何被AOP代理,包括代理对象的创建、配置与执行逻辑,帮助读者全面掌握Spring AOP的核心技术。
43 3
Spring高手之路23——AOP触发机制与代理逻辑的执行
|
1月前
|
Java Spring
[Spring]aop的配置与使用
本文介绍了AOP(面向切面编程)的基本概念和核心思想。AOP是Spring框架的核心功能之一,通过动态代理在不修改原代码的情况下注入新功能。文章详细解释了连接点、切入点、通知、切面等关键概念,并列举了前置通知、后置通知、最终通知、异常通知和环绕通知五种通知类型。
30 1
|
1月前
|
XML Java 开发者
论面向方面的编程技术及其应用(AOP)
【11月更文挑战第2天】随着软件系统的规模和复杂度不断增加,传统的面向过程编程和面向对象编程(OOP)在应对横切关注点(如日志记录、事务管理、安全性检查等)时显得力不从心。面向方面的编程(Aspect-Oriented Programming,简称AOP)作为一种新的编程范式,通过将横切关注点与业务逻辑分离,提高了代码的可维护性、可重用性和可读性。本文首先概述了AOP的基本概念和技术原理,然后结合一个实际项目,详细阐述了在项目实践中使用AOP技术开发的具体步骤,最后分析了使用AOP的原因、开发过程中存在的问题及所使用的技术带来的实际应用效果。
61 5
|
1月前
|
JSON Java 数据库
SpringBoot项目使用AOP及自定义注解保存操作日志
SpringBoot项目使用AOP及自定义注解保存操作日志
43 1
|
1月前
|
安全 Java 测试技术
Java开发必读,谈谈对Spring IOC与AOP的理解
Spring的IOC和AOP机制通过依赖注入和横切关注点的分离,大大提高了代码的模块化和可维护性。IOC使得对象的创建和管理变得灵活可控,降低了对象之间的耦合度;AOP则通过动态代理机制实现了横切关注点的集中管理,减少了重复代码。理解和掌握这两个核心概念,是高效使用Spring框架的关键。希望本文对你深入理解Spring的IOC和AOP有所帮助。
35 0
|
2月前
|
Java 编译器 Spring
Spring AOP 和 AspectJ 的区别
Spring AOP和AspectJ AOP都是面向切面编程(AOP)的实现,但它们在实现方式、灵活性、依赖性、性能和使用场景等方面存在显著区别。‌
95 2
|
2月前
|
Java Spring 容器
Spring IOC、AOP与事务管理底层原理及源码解析
【10月更文挑战第1天】Spring框架以其强大的控制反转(IOC)和面向切面编程(AOP)功能,成为Java企业级开发中的首选框架。本文将深入探讨Spring IOC和AOP的底层原理,并通过源码解析来揭示其实现机制。同时,我们还将探讨Spring事务管理的核心原理,并给出相应的源码示例。
137 9
|
2月前
|
Java 容器
AOP面向切面编程
AOP面向切面编程
43 0