当前位置: 首页 > news >正文

Spring 学习(二)AOP

一、什么是AOP

Aspect Oriented Programming,即面向切面编程。对一个大型项目的代码而言,整个系统要求关注安全检查、日志、事务等功能,这些功能实际上“横跨”多个业务方法。在一般的OOP编程里,需要在每一个业务方法内添加相关非业务方法的调用,这实际上是冗余的。如果能够类似IoC一样,这样的安全检查,日志,事务功能单独提取放到外面,核心业务方法不需要关注,就能降低代码耦合度。如果我们以AOP的视角来编写上述业务,可以依次实现:

  • 核心逻辑Service
  • 切面逻辑,即:
    • 权限检查的Aspect;
    • 日志的Aspect;
    • 事务的Aspect。
      然后,以某种方式,让框架来把上述3个Aspect以Proxy的方式“织入”到Service中,这样一来,就不必编写复杂而冗长的Proxy模式

AOP对于解决特定问题,例如事务管理非常有用,这是因为分散在各处的事务代码几乎是完全相同的,并且它们需要的参数(JDBC的Connection)也是固定的。另一些特定问题,如日志,就不那么容易实现,因为日志虽然简单,但打印日志的时候,经常需要捕获局部变量,如果使用AOP实现日志,我们只能输出固定格式的日志,因此,使用AOP时,必须适合特定的场景。


二、AOP

常见的AOP术语包括:

  • 切面(Aspect):横跨多个类和方法的模块,定义了一组横切关注点的行为。
  • 连接点(Join Point):程序执行过程中可以插入切面的特定点,例如方法调用、异常抛出等。
  • 通知(Advice):切面在连接点上执行的行为,包括前置通知、后置通知、环绕通知、异常通知和最终通知等。
  • 切点(Pointcut):定义了一组连接点的表达式,用于确定切面在哪些连接点上执行。
  • 引入(Introduction):允许向现有类添加新的方法或属性。
  • 织入(Weaving):将切面应用到目标对象中,以创建新的代理对象或修改现有对象的字节码。

例如,编写日志loggin切片

@Aspect
@Component
public class LoggingAspect {// 在执行UserService的每个方法前执行:@Before("execution(public * com.itranswarp.learnjava.service.UserService.*(..))")public void doAccessCheck() {System.err.println("[Before] do access check...");}// 在执行MailService的每个方法前后执行:@Around("execution(public * com.itranswarp.learnjava.service.MailService.*(..))")public Object doLogging(ProceedingJoinPoint pjp) throws Throwable {System.err.println("[Around] start " + pjp.getSignature());Object retVal = pjp.proceed();System.err.println("[Around] done " + pjp.getSignature());return retVal;}
}
  • @Before()里面的字符串是告诉AspectJ应该在何处执行该方法,这里写的意思是:执行UserService的每个public方法前执行doAccessCheck()代码
  • @Around注解,它和@Before不同,@Around可以决定是否执行目标方法
  • @Aspect注解,表示它的@Before标注的方法需要注入到UserService的每个public方法执行前,@Around标注的方法需要注入到MailService的每个public方法执行前后

我们需要给@Configuration类加上一个@EnableAspectJAutoProxy注解,Spring的IoC容器看到这个注解,就会自动查找带有@Aspect的Bean,然后根据每个方法的@Before、@Around等注解把AOP注入到特定的Bean中。

@Configuration
@ComponentScan
@EnableAspectJAutoProxy
public class AppConfig {...
}

三、AOP实现原理

AOP原理实际上就是一个代理类的实现,这个代理是借由Spring容器实现的,例如,将LoggingAspect.doAccessCheck注入UserService的每个public方法中,可以通过如下实现:

public UserServiceAopProxy extends UserService {private UserService target;private LoggingAspect aspect;public UserServiceAopProxy(UserService target, LoggingAspect aspect) {this.target = target;this.aspect = aspect;}public User login(String email, String password) {// 先执行Aspect的代码:aspect.doAccessCheck();// 再执行UserService的逻辑:return target.login(email, password);}public User register(String email, String password, String name) {aspect.doAccessCheck();return target.register(email, password, name);}...
}

容器自动为我们创建了注入了Aspect的子类,取代原始的UserService,并把被@Before @Around关键词修饰的方法覆写。


四、拦截器

AOP拦截器通常是指用于拦截和处理方法调用的组件或类。这些拦截器可以在方法调用前、后或异常抛出时执行特定的逻辑

  • @Before:该注解用于定义前置拦截器。在目标方法执行前,被注解的方法将被执行。可以用于实现预处理、参数验证和权限检查等功能。
  • @AfterReturning:该注解用于定义后置拦截器。在目标方法成功执行并返回结果后,被注解的方法将被执行。可以用于实现日志记录、结果处理和清理操作等功能。
  • @Around:该注解用于定义环绕拦截器。在目标方法执行前后,被注解的方法将被执行。通过在拦截器方法中调用ProceedingJoinPoint.proceed()来控制目标方法的执行,并可以在适当的时机添加额外的逻辑。
  • @AfterThrowing:
    该注解用于定义异常拦截器。在目标方法抛出异常时,被注解的方法将被执行。可以用于实现异常处理、错误日志记录等功能。

通过使用这些注解,可以将拦截器逻辑与特定的切点(Pointcut)相结合,实现对核心业务逻辑的拦截和处理。可以使用execution()表达式来定义切点,指定要拦截的方法的执行。这些注解可以与Spring AOP的其他功能和配置一起使用,如切面(Aspect)、通知(Advice)和配置文件等,以实现更复杂的AOP编程。


五、注解装配

使用注解装配AOP时,最好需要在被装配的Bean处使用注解标记,被装配的Bean最好自己能清清楚楚地知道自己被安排了。例如,Spring提供的@Transactional。

我们以一个实际例子演示如何使用注解实现AOP装配。为了监控应用程序的性能,我们定义一个性能监控的注解:

@Target(METHOD)
@Retention(RUNTIME)
public @interface MetricTime {String value();
}/**************************************/
@Component
public class UserService {// 监控register()方法性能:@MetricTime("register")public User register(String email, String password, String name) {...}...
}/**************************************/
@Aspect
@Component
public class MetricAspect {@Around("@annotation(metricTime)")public Object metric(ProceedingJoinPoint joinPoint, MetricTime metricTime) throws Throwable {String name = metricTime.value();long start = System.currentTimeMillis();try {return joinPoint.proceed();} finally {long t = System.currentTimeMillis() - start;// 写入日志或发送至JMX:System.err.println("[Metrics] " + name + ": " + t + "ms");}}
}

这段代码中,一直对@Around("@annotation(metricTime)")存在疑惑,为什么不是MetricTime(类)而是metricTime(参数名),关键在@annotation。

@annotation 是 Spring AOP 中的一个切点指示器,用于匹配被任意注解标记的方法或类。在切点表达式中使用 @annotation(annotationType),其中 annotationType 是要匹配的注解类型,可以是任何有效的注解类型。这个切点指示器的作用是,在切面中匹配被特定注解标记的方法或类,以便在相应的切面方法中对它们进行处理。

以下实例中,@Around(“@annotation(com.example.MyAnnotation)”) 表达式表示匹配被 @MyAnnotation 注解标记的方法。当调用这些被标记的方法时,切面中的 myAdvice 方法会被触发

@Aspect
@Component
public class MyAspect {// 匹配被 @MyAnnotation 标记的方法@Around("@annotation(com.example.MyAnnotation)")public Object myAdvice(ProceedingJoinPoint joinPoint) throws Throwable {// 在方法执行前后执行自定义逻辑// ...return joinPoint.proceed();}}

但在@Around("@annotation(metricTime)")中,由于切面方法metric要用到被标记注解的实例,而不只是匹配注解类型。那么在 @annotation() 中需要传入该注解的实例。具体来说,如果被切入的方法的参数列表中有一个具有某个特定注解的参数,则可以在切面表达式中使用 @annotation(parameterName) 来匹配并获取该注解的实例。

http://www.lryc.cn/news/166905.html

相关文章:

  • 笔记1.1 计算机网络基本概念
  • 液压切管机配套用液压泵站比例阀放大器
  • C++ Primer Plus 第七章笔记
  • 常用数据库的 API - 开篇
  • C++之生成详细汇编代码(二百一十六)
  • AIGC|当一个程序员学会用AI来辅助编程实践
  • 9.14号作业
  • 【面试题】C/C++ 中指针和引用的区别
  • spring boot 整合多数据源
  • 数据集成:数据挖掘的准备工作之一
  • xml配置文件密码特殊字符处理
  • 遥感数据与作物模型同化
  • UI库DHTMLX Suite v8.2发布全新表单组件,让Web表单实现高度可定制!
  • 河北省图书馆典藏《乡村振兴振兴战略下传统村落文化旅游设计》许少辉八一新著
  • 什么是卷积002
  • 黑马JVM总结(八)
  • 开源网安入选广东省网络空间安全标准化技术委员会新技术及应用安全技术工作组成员单位
  • Nginx配置指南:如何定位、解读与优化Linux上的Nginx设置
  • 辉瑞与吉利德科学:制药巨头的新冠病毒之战
  • x86架构基础汇编知识
  • ThreadLocal的原理
  • Chrome 108版(64-bit 108.0.5359.125)网盘下载
  • Mars3d插件参考开发教程并在相关页面引入
  • Windows 性能突然打鸡血,靠 Bug 修复了多年顽疾
  • 亚马逊封买家账号的原因有哪些
  • 1.0零基础尝试DCM通讯(c-store)
  • vue之封装tab类组件
  • 固定资产管理中净值怎么算
  • SQlite操作后如何正确退出
  • phpcmsV9.6.0sql注入漏洞分析