Spring A0P原理、基本概念及示例

本文介绍了面向切面编程(AOP)的基本概念及其在Spring框架中的应用。详细解释了AOP如何通过分离业务逻辑与系统服务提高代码内聚性,并通过示例展示了如何使用Spring AOP实现日志记录等功能。
一、原理
   AOP即Aspect-OrientedProgramming的缩写,中文意思是面向切面(或方面)编程。AOP实际上是一种编程思想,可以通过预编译方式和运行期动态代理实现在不修改源代码的情况下给程序动态统一添加功能的一种思想。

   在传统的面向对象(Object-Oriented Programming,OOP)编程中,我们总是按照某种特定的执行顺序来实现业务流程,各个执行步骤之间是相互衔接、相互耦合的,对垂直切面关注度很高,横切面关注却很少,也很难关注。那么怎样可以解决这个问题呢?我们需要AOP,关注系统的“切面”,在适当的时候“拦截”程序的执行流程,把程序的预处理和后处理交给某个拦截器来完成。这样,业务流程就完全的从其它无关的代码中解放出来,各模块之间的分工更加明确,程序维护也变得容易多了。

  正如上所说,AOP不是一种技术,而是编程思想。凡是符合AOP思想的技术,都可以看成是AOP的实现。目前的AOP实现有AspectJ、JBoss4.0、nanning、spring等项目。其中Spring对AOP进行了很好的实现,同时SpringAOP也是Spring的两大核心一。

   AOP的发展目前已经历了两个阶段:第一阶--静态AOP和第二阶段—动态AOP。静态AOP阶段,相应的横切关注点以Aspect形式实现之后,会通过特定的编译器,将实现后的Aspect编译并织入到系统的静态类中。比如AspectJ会使用ajc编译器将各个Aspect以Java字节码的形式编译到系统的各个功能模块中,已达到融合Aspect和Class的目的。动态AOP阶段,AOP的织入过程在系统运行开始之后进行,而不是预先编译到系统中,可以在调整织入点以及织入逻辑单元的同时,不必变更系统其他的模块,甚至在系统运行的时候,也可以动态更改织入逻辑。这两个阶段也为我们提供了通过预编译方式和运行期动态代理方式,实现了在不修改源代码的情况下给程序统一添加功能的目的。

   AOP主要的意图是:允许通过分离应用的业务逻辑与系统级服务进行内聚性的开发。应用对象只实现业务逻辑即可,并不负责其它的系统级关注点。

   AOP主要应用场景是:日志记录、跟踪、监控和优化,性能统计、优化,安全、权限控制,应用系统的异常捕捉及处理,事务处理,缓存,持久化,懒加载(Lazyloading),内容传递,调试,资源池,同步等等。


二、基本概念

切面(Aspect) :官方的抽象定义为“一个关注点的模块化,这个关注点可能会横切多个对象”

连接点(Joinpoint):程序执行过程中明确的点,如方法的调用或特定的异常被抛出。

通知(Advice):在特定的连接点,AOP框架执行的动作。各种类型的通知包括“around”、“before”和“throws”通知。通知类型将在下面讨论。许多AOP框架包括Spring都是以拦截器做通知模型,维护一个“围绕”连接点的拦截器链。

切入点(Pointcut):指定一个通知将被引发的一系列连接点的集合。AOP框架必须允许开发者指定切入点,例如,使用正则表达式。例如:execution(*com.spring.service.*.*(..))

引入(Introduction):匹配连接点的断言,在AOP中通知和一个切入点表达式关联。添加方法或字段到被通知的类。Spring允许引入新的接口到任何被通知的对象。例如,你可以使用一个引入使任何对象实现IsModified接口,来简化缓存。

目标对象(TargetObject):被一个或者多个切面所通知的对象,包含连接点的对象,也被称作被通知或被代理对象。

AOP代理(AOPProxy):AOP框架创建的对象,包含通知。在Spring中,AOP代理可以是JDK动态代理或CGLIB代理。

Around通知:包围一个连接点的通知,如方法调用。这是最强大的通知。Aroud通知在方法调用前后完成自定义的行为,它们负责选择继续执行连接点或通过返回它们自己的返回值或抛出异常来短路执行。

Before通知:在一个连接点之前执行的通知,但这个通知不能阻止连接点前的执行(除非它抛出一个异常)。

Throws通知:在方法抛出异常时执行的通知。Spring提供强制类型的Throws通知,因此你可以书写代码捕获感兴趣的异常(和它的子类),不需要从Throwable或Exception强制类型转换。

Afterreturning通知:在连接点正常完成后执行的通知,例如,一个方法正常返回,没有抛出异常。


三、示例

1、使用注解的配置文件 applicationContext-aop.xml



   xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:aop="http://www.springframework.org/schema/aop"
   xmlns:context="http://www.springframework.org/schema/context"
   xsi:schemaLocation="http://www.springframework.org/schema/beans
            http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
            http://www.springframework.org/schema/context
            http://www.springframework.org/schema/context/spring-context-3.0.xsd
            http://www.springframework.org/schema/aop 
              http://www.springframework.org/schema/aop/spring-aop-3.0.xsd">
   
   
   
   
   
   

2、定义一个切面

@Aspect
@Component
public class AnnoAsect {
   
   
   @Pointcut("execution(* com.user.service.*.*(..))")
    public voidanyMethod(){}
   
   @Around("execution(* com.user.service.*.*(..))")
    publicObject doAround(ProceedingJoinPoint pjp)throws Throwable{
       //环绕通知(Around advice):包围一个连接点的通知
       //类似Web中Servlet规范中的Filter的doFilter方法。可以在方法的调用前后完成自定义的行为,
      //也可以选择不执行。ApplicationContext中在里面使用元素进行声明。
      //例如,TestAspect中的doAround方法。
      
      //执行真正调用的方法
       ObjectretVal = pjp.proceed();
       returnretVal;
      
    }
   
   @Before(value="anyMethod()")
    public voiddobefore(JoinPoint jp){
       //前置通知(Before advice):在某连接点(JoinPoint)之前执行的通知
       //这个通知不能阻止连接点前的执行。ApplicationContext中在里面使用
      //元素进行声明。例如,TestAspect中的doBefore方法
      System.out.println("前置通知!");
      //获取请求参数
       Object[]args = jp.getArgs();
      if(args.length > 0){
          for (Objectobject : args) {
            System.out.println("参数[" + object + "]");
             Useruser=(User)object;
            if("".equals(user.getUsername())){
               System.out.println("---------------");
             }
          }
       }
    }
   
   @After("anyMethod()")
    public voiddoAfter(JoinPoint jp) {
       // 后通知(Afteradvice):当某连接点退出的时候执行的通知(不论是正常返回还是异常退出)。
       //ApplicationContext中在里面使用元素进行声明。例如,TestAspect中的doAfter方法,所以AOPTest中调用BServiceImpl.barB抛出异常时,doAfter方法仍然执行
      System.out.println("后通知(After advice)类型");
      
    }

   @AfterReturning(pointcut="anyMethod()")
    public voiddoAfterReturn(JoinPoint jp) {
       //返回后通知(After return advice):在某连接点正常完成后执行的通知
       //不包括抛出异常的情况。ApplicationContext中在里面使用元素进行声明。
      System.out.println("返回后通知(After return advice)类型");
      
    }

   @AfterThrowing(throwing="ex", pointcut="anyMethod()")
    public voiddoThrowing(JoinPoint jp, Throwable ex) {
       //抛出异常后通知(After throwing advice) : 在方法抛出异常退出时执行的通知。
       //ApplicationContext中在里面使用元素进行声明。例如,TestAspect中的doThrowing方法。
      System.out.println("方法抛出异常后到了这里抛出异常");
      
    }
}



<think>好的,我需要帮助用户使用AOP实现方法前后的日志记录功能。首先,我需要回忆一下AOP的基本概念和相关术语。根据用户提供的引用内容,AOP是面向切面编程,通过切面提取业务处理中的共同逻辑,比如日志记录,这样可以降低耦合度,提高可维护性。用户提到在Spring框架中使用AOP,所以应该基于Spring AOP来实现。 接下来,我需要确定实现方法前后日志记录的具体步骤。通常,AOP中的切面(Aspect)由切点(Pointcut)和通知(Advice)组成。切点定义了在哪些方法上应用通知,而通知则是在方法执行的不同阶段插入的代码,比如@Before、@After、@Around等注解。 用户想要记录方法执行前和执行后的日志,所以应该使用@Around通知,因为它可以在方法执行前后都插入代码。或者,也可以结合@Before和@After通知来实现。不过,@Around更灵活,可以控制方法是否执行,并获取执行时间等信息。 然后,我需要考虑如何定义切点表达式。例如,使用execution表达式来匹配需要记录日志的方法。比如,execution(* com.example.service.*.*(..))表示匹配com.example.service包下所有类的所有方法。 接下来,需要编写切面类,使用@Aspect注解,并在其中定义通知方法。例如,使用@Around注解的方法,方法参数为ProceedingJoinPoint,通过调用proceed()来执行目标方法,并在其前后记录日志。 另外,用户可能需要记录方法的参数、返回结果、执行时间等信息。这时候可以在通知方法中获取joinPoint的参数,记录到日志中。同时,处理可能出现的异常,可以使用@AfterThrowing来记录异常情况。 还要注意在Spring配置中启用AOP支持,比如使用@EnableAspectJAutoProxy注解,或者XML配置方式。 根据用户提供的引用[1]和[2],Spring AOP使用的是动态代理,对于接口实现类使用JDK动态代理,对于没有接口的类使用CGLIB代理。这点可能需要提醒用户,确保目标类符合代理的要求。 另外,可能需要考虑日志记录的性能影响,比如异步记录日志,但用户当前的需求可能暂时不需要,保持简单即可。 最后,应该提供一个代码示例,展示如何实现这个切面,包括切面类、切点定义、通知方法,以及如何在Spring中启用AOP。同时,可能需要解释每个部分的作用,帮助用户理解。 总结步骤: 1. 添加Spring AOP依赖。 2. 创建切面类,使用@Aspect和@Component注解。 3. 定义切点表达式,匹配目标方法。 4. 编写@Around通知方法,记录方法执行前后的日志,包括参数、结果、执行时间。 5. 处理异常情况,可选添加@AfterThrowing通知。 6. 启用AOP支持,确保切面生效。 7. 测试日志记录功能,验证是否正确输出。 在编写代码时,要注意LaTeX格式,比如方法名使用$...$包裹,如@Around("logPointcut()")。同时,确保代码示例正确,语法无误,并且符合用户的中文要求。</think>要使用AOP实现方法执行前后的日志记录功能,可以按照以下步骤实现(以Spring AOP为例): ### 一、添加依赖 ```xml <!-- Spring AOP 依赖 --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-aop</artifactId> </dependency> ``` ### 二、定义日志切面类 ```java @Aspect @Component public class LogAspect { // 定义切点(匹配目标方法) @Pointcut("execution(* com.example.service.*.*(..))") public void logPointcut() {} // 环绕通知(记录方法执行前后) @Around("logPointcut()") public Object logAround(ProceedingJoinPoint joinPoint) throws Throwable { String methodName = joinPoint.getSignature().getName(); Object[] args = joinPoint.getArgs(); // 方法执行前日志 System.out.println("方法执行前:" + methodName + ",参数:" + Arrays.toString(args)); long startTime = System.currentTimeMillis(); try { Object result = joinPoint.proceed(); // 执行目标方法 long endTime = System.currentTimeMillis(); // 方法执行后日志 System.out.println("方法执行后:" + methodName + ",耗时:" + (endTime - startTime) + "ms" + ",结果:" + result); return result; } catch (Exception e) { System.out.println("方法异常:" + methodName + ",异常信息:" + e.getMessage()); throw e; } } } ``` ### 三、关键术语说明 1. **切面(Aspect)**:通过`@Aspect`注解定义,包含多个切点和通知 2. **切点(Pointcut)**:通过`@Pointcut`表达式匹配需要增强的方法 3. **通知(Advice)**:包含`@Around`(环绕)、`@Before`(前置)、`@After`(后置)等类型 ### 四、实现原理 Spring AOP通过动态代理技术实现[^2]: - 对接口实现类使用**JDK动态代理** - 对非接口类使用**CGLIB代理** - 代理对象会在目标方法执行前后插入日志逻辑 ### 五、优化建议 1. 使用`SLF4J`代替`System.out`进行规范日志输出 2. 添加`@AfterThrowing`注解处理异常日志 3. 通过异步方式记录日志提升性能 ```java // 异常日志记录示例 @AfterThrowing(pointcut = "logPointcut()", throwing = "ex") public void logException(JoinPoint joinPoint, Exception ex) { System.out.println("异常方法:" + joinPoint.getSignature().getName()); System.out.println("异常堆栈:" + Arrays.toString(ex.getStackTrace())); } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值