ToB企服应用市场:ToB评测及商务社交产业平台

标题: spring AOP总结 [打印本页]

作者: 王海鱼    时间: 2023-12-5 06:25
标题: spring AOP总结
一、概念

  AOP面向切面编程,一种编程范式
二、作用

  在不改动原始设计(原代码不改动)的基础上为方法进行功能增强(即增加功能)
三、核心概念

  1、代理(Proxy):SpringAOP的核心本质是采用代理模式实现的
  2、连接点(JoinPoint):在SpringAOP中,理解为任意方法的执行
  3、切入点(Pointcut):匹配连接点的式子,也是具有共性功能的方法描述
  4、通知(Advice):若干个方法的共性功能,在切入点处执行,最终体现为一个方法
  5、切面(Aspect):描述通知与切入点的对应关系
  6、目标对象(Target):被代理的原始对象成为目标对象
四、快速开始

  1、导入相关依赖
  由于导入spring-context时会自动导入spring的AOP包所以,这里只用导入aspectjweaver即可。
  在pom.xml文件中导入
 
  1.     <dependencies><br>   
  2.         <dependency>
  3.             <groupId>org.springframework</groupId>
  4.             <artifactId>spring-context</artifactId>
  5.             <version>5.3.29</version>
  6.         </dependency><br>   
  7.         <dependency>
  8.             <groupId>org.aspectj</groupId>
  9.             <artifactId>aspectjweaver</artifactId>
  10.             <version>1.9.19</version>
  11.         </dependency>
  12.     </dependencies>
复制代码
 
  2、定义dao接口与实现类
  在dao下面创建BookDao接口类文件
  1. public interface BookDao {
  2.     public void save();
  3.     public void update();
  4. }
复制代码
  在dao下面创建impl文件夹,里面创建BookDao的实现类BookDaoImpl
  1. package com.itheima.dao.impl;
  2. import com.itheima.dao.BookDao;
  3. import org.springframework.stereotype.Repository;
  4. @Repository
  5. public class BookDaoImpl implements BookDao {
  6.     public void save() {
  7.         System.out.println(System.currentTimeMillis());
  8.         System.out.println("book dao save ...");
  9.     }
  10.     public void update(){
  11.         System.out.println("book dao update ...");
  12.     }
  13. }
复制代码
  3、定义通知类,制作通知方法
  创建aop文件夹,并在下面创建类MyAdvice。
  1. package com.itheima.aop;
  2. import org.springframework.stereotype.Component;
  3. @Component
  4. public class MyAdvice {
  5.     public void method() {
  6.         System.out.println(System.currentTimeMillis());
  7.         System.out.println("进行增强功能");
  8.     }
  9. }
复制代码
  4、定义切入点表达式、配置切面(绑定切入点与通知关系)
  1. package com.itheima.aop;<br><br>import org.aspectj.lang.annotation.Aspect;<br>import org.aspectj.lang.annotation.Before;<br>import org.aspectj.lang.annotation.Pointcut;<br>import org.springframework.stereotype.Component;<br><br>@Component<br>@Aspect<br>public class MyAdvice {<br>    //设置切入点,@Pointcut注解要求配置在方法上方<br>    @Pointcut("execution(void com.itheima.dao.BookDao.update())")<br>    private void pt(){}<br>    //设置在切入点pt()的前面运行当前操作(前置通知)<br>    @Before("pt()")<br>    public void method() {<br>        System.out.println(System.currentTimeMillis());<br>        System.out.println("进行增强功能");<br>    }<br>}
复制代码
  5、在配置类中进行Spring注解包扫描和开启AOP功能  
  创建config文件夹,创建Spring的配置文件。
  1. package com.itheima.config;
  2. import org.springframework.context.annotation.ComponentScan;
  3. import org.springframework.context.annotation.Configuration;
  4. import org.springframework.context.annotation.EnableAspectJAutoProxy;
  5. // 告知spring这是个配置类
  6. @Configuration
  7. // 扫描指定包
  8. @ComponentScan(basePackages = {"com.itheima"})
  9. // 开启aop,告知spring开启使用注解的方式开启aop
  10. @EnableAspectJAutoProxy
  11. public class SpringConfig {
  12. }
复制代码
  6、创建启动文件
  在项目目录下创建启动类文件,App文件
  1. package com.itheima;
  2. import com.itheima.config.SpringConfig;
  3. import com.itheima.dao.BookDao;
  4. import org.springframework.context.annotation.AnnotationConfigApplicationContext;
  5. public class App {
  6.     public static void main(String[] args) {
  7.         // 1. 创建容器对象, 传入配置类
  8.         AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(SpringConfig.class);
  9.         // 2. 从容器中获取对象
  10.         BookDao bookDao = context.getBean(BookDao.class);
  11.         // 3. 执行方法
  12. //        bookDao.save();
  13.         bookDao.update();
  14.     }
  15. }
复制代码
  7、最终项目目录结构

 
五、切入点表达式

  1、切入点:要进行增强的方法
  2、切入点表达式:要进行增强的方法的描述方式
    切入点表达式标准格式:动作关键字(访问修饰符   返回值   包名.类/接口名.方法名(参数)异常名)
  1. execution(* com.testweb.service.*Service.*(..))
复制代码
  切入点表达式描述通配符
    作用:用于快速描述,范围描述
    * :匹配任意符号(常用)
    ..:匹配多个连续的任意符号(常用)
    +:匹配子类型
  切入点表达式书写技巧
    1、按标准规范开发
    2、查询操作的返回值建议使用*匹配
    3、减少使用 .. 的形式描述包
    4、对接口进行描述,使用 * 表示模块名,例如UserService的匹配描述为 *Service 
    5、方法名书写保留动词,例如get,使用 * 表示名词,例如getById匹配描述为 getBy*
    6、参数根据实际情况灵活调整
 六、通知类型

  1、通知类型
    1、前置通知
    2、后置通知
    3、环绕通知(重点)
      *  环绕通知依赖形参ProceedingJoinPoint才能实现对原始方法的调用
      *  环绕通知可以隔离原始方法的调用执行
      *  环绕通知返回值设置为object类型
      *  环绕通知中可以对原始方法调用过程中出现的异常进行处理
    4、返回后通知
    5、抛出异常后通知
  2、AOP通知获取数据
    1、获取切入点方法的参数
      *  JoinPoint:适用于前置(@Before)、后置(@after)、返回后(@AfterReturning)、抛出异常后通知(@AfterThrowing)
  1. package com.itheima.aop;
  2. import org.aspectj.lang.JoinPoint;
  3. import org.aspectj.lang.annotation.Aspect;
  4. import org.aspectj.lang.annotation.Before;
  5. import org.aspectj.lang.annotation.Pointcut;
  6. import org.springframework.stereotype.Component;
  7. @Component
  8. @Aspect
  9. public class MyAdvice {
  10.     //设置切入点,@Pointcut注解要求配置在方法上方
  11.     @Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")
  12.     private void pt(){}
  13.     //设置在切入点pt()的前面运行当前操作(前置通知)
  14.     @Before("pt()")
  15.     public void method(JoinPoint jp) {
  16.         // 获取参数
  17.         for (Object o : jp.getArgs()) {
  18.             System.out.println(o);
  19.         }
  20.         System.out.println(System.currentTimeMillis());
  21.         System.out.println("进行增强功能");
  22.     }
  23. }
复制代码
      * ProceedJointPoint:适用于环绕通知
  1. package com.itheima.aop;
  2. import org.aspectj.lang.ProceedingJoinPoint;
  3. import org.aspectj.lang.annotation.*;
  4. import org.springframework.stereotype.Component;
  5. @Component
  6. @Aspect
  7. public class MyAdvice {
  8.     //设置切入点,@Pointcut注解要求配置在方法上方
  9.     @Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")
  10.     private void pt() {
  11.     }
  12.     //设置在切入点pt()的前面运行当前操作(前置通知)
  13.     @Around("pt()")
  14.     public Object method(ProceedingJoinPoint pjp) throws Throwable {
  15.         // 获取参数
  16.         Object[] args = pjp.getArgs();
  17.         // 修改原先的参数
  18.         args[0] = 666;
  19.         // 执行原方法,并传入参数
  20.         Object ret = pjp.proceed(args);
  21.         System.out.println("进行增强功能");
  22.         return ret;
  23.     }
  24. }
复制代码
    2、获取切入点方法返回值
      *  返回后通知
  1. package com.itheima.aop;
  2. import org.aspectj.lang.JoinPoint;
  3. import org.aspectj.lang.ProceedingJoinPoint;
  4. import org.aspectj.lang.annotation.*;
  5. import org.springframework.stereotype.Component;
  6. import java.util.Arrays;
  7. @Component
  8. @Aspect
  9. public class MyAdvice {
  10.     //设置切入点,@Pointcut注解要求配置在方法上方
  11.     @Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")
  12.     private void pt() {
  13.     }
  14.     // 设置在原函数执行后执行当前操作(后置通知)
  15.     // 如果有返回值,则可以写成afterReturning(value = "pt()", returning = "ret"),使用ret接收原方法返回值
  16.     // 如果有参数,则public void afterReturning(JoinPoint jp, Object ret),且JoinPoint jp,一定要在前面
  17.     @AfterReturning(value = "pt()", returning = "ret")
  18.     public void afterReturning(JoinPoint jp, Object ret) {
  19.         // 获取参数
  20.         Object[] args = jp.getArgs();
  21.         System.out.println("参数:" + Arrays.toString(args));
  22.         System.out.println("返回值:" + ret);
  23.         System.out.println("执行后置通知");
  24.     }
  25. }
复制代码
      * 环绕通知
  1. package com.itheima.aop;
  2. import org.aspectj.lang.ProceedingJoinPoint;
  3. import org.aspectj.lang.annotation.*;
  4. import org.springframework.stereotype.Component;
  5. @Component
  6. @Aspect
  7. public class MyAdvice {
  8.     //设置切入点,@Pointcut注解要求配置在方法上方
  9.     @Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")
  10.     private void pt() {
  11.     }
  12.     //设置在切入点pt()的前面运行当前操作(前置通知)
  13.     @Around("pt()")
  14.     public Object method(ProceedingJoinPoint pjp) throws Throwable {
  15.         // 获取参数
  16.         Object[] args = pjp.getArgs();
  17.         // 修改原先的参数
  18.         args[0] = 666;
  19.         // 执行原方法,并传入参数,并用ret接收原方法返回值
  20.         Object ret = pjp.proceed(args);
  21.         System.out.println("进行增强功能");
  22.         return ret;
  23.     }
  24. }
复制代码
    3、获取切入点方法运行异常信息
      * 抛出异常后通知
  1. package com.itheima.aop;
  2. import org.aspectj.lang.JoinPoint;import org.aspectj.lang.annotation.*;
  3. import org.springframework.stereotype.Component;
  4. import java.util.Arrays;
  5. @Component
  6. @Aspect
  7. public class MyAdvice {
  8.     //设置切入点,@Pointcut注解要求配置在方法上方
  9.     @Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")
  10.     private void pt() {
  11.     }<br><br>
  12.     @AfterThrowing(value = "pt()", throwing = "e")
  13.     public void afterThrowing(JoinPoint jp, Exception e) {
  14.         // 获取参数
  15.         Object[] args = jp.getArgs();
  16.         System.out.println("参数:" + Arrays.toString(args));
  17.         System.out.println("异常:" + e);
  18.         System.out.println("执行异常后置通知");
  19.     }
  20. }
复制代码
      * 环绕通知
  1. package com.itheima.aop;
  2. import org.aspectj.lang.ProceedingJoinPoint;
  3. import org.aspectj.lang.annotation.*;
  4. import org.springframework.stereotype.Component;
  5. @Component
  6. @Aspect
  7. public class MyAdvice {
  8.     //设置切入点,@Pointcut注解要求配置在方法上方
  9.     @Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")
  10.     private void pt() {
  11.     }//设置在切入点pt()的前面运行当前操作(前置通知)
  12.     @Around("pt()")
  13.     public Object method(ProceedingJoinPoint pjp) {
  14.         // 获取参数
  15.         Object[] args = pjp.getArgs();
  16.         // 修改原先的参数
  17.         args[0] = 666;
  18.         // 执行原方法,并传入参数
  19.         Object ret = null;<br>     // 获取异常
  20.         try {
  21.             ret = pjp.proceed(args);
  22.         }catch (Throwable e) {
  23.             e.printStackTrace();
  24.         }
  25.         System.out.println("进行增强功能");
  26.         return ret;
  27.     }
  28. }
复制代码
 

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!




欢迎光临 ToB企服应用市场:ToB评测及商务社交产业平台 (https://dis.qidao123.com/) Powered by Discuz! X3.4