AOP概念理解

AOP概念理解

什么是AOP?

AOP称为面向切面编程,用于将那些与业务无关,但却对多个对象产生影响的公共行为和逻辑,抽取并封装为一个可重用的模块,这个模块被命名为“切面”( Aspect ),以切面作为基本单位。通过代理的方式,在调用想用的对象方法时进行拦截处理,执行切入的逻辑,然后再调用真正的方法实现。减少系统中的重复代码,降低了模块间的耦合度,同时提高了系统的可维护性。

概念组成

来自黑马程序员

  • 连接点:在程序执行期间可以插入切面的点。
  • 切入点:多个连接点的集合。
  • 通知/增强:在切面的某个特定连接点上执行的增强(拓展的功能)。五种通知类型:
    前置、后置、环绕、返回通知、异常通知。
  • 切面 :切面是一个类(@Aspect注解声明)。将与业务无关,但却对多个对象产生影响的公共行为和逻辑,抽取并封装为一个可重用的模块。
  • 织入:将切面应用到目标对象并创建代理对象的过程。

应用场景

  • 记录操作日志
  • 权限管理
  • Spring内置的事务处理

实现方式

AOP有两种实现方式:静态代理和动态代理。

  • 静态代理:在编译时或类加载时进行切面的织入,AspectJ就是静态代理。
  • 动态代理:在程序运行时进行切面的织入,AOP框架不会去修改字节码,而是在内存中临时生成一个代理对象,在运行期间对业务方法进行增强,不会生成新类。
    动态代理主要有两种实现方式:
  • JDK 动态代理:JDK 动态代理要求被代理的类必须实现一个接口,它通过反射来接收被代理的类,并使用 InvocationHandler 接口和 Proxy 类实现代理。
  • CGLIB 动态代理:CGLIB 则是一个代码生成的类库,它可以在运行时动态地生成某个类的子类,通过继承的方式实现代理。如果目标类没有实现接口,Spring AOP 会选择使用 CGLIB 来动态代理目标类。
### Java 中 AOP概念详解 AOP(Aspect-Oriented Programming,面向方面编程)是一种用于增强模块化设计的技术[^1]。它的主要目标是解决传统面向对象编程中难以处理的横切关注点问题。所谓横切关注点是指那些无法被封装到单一类或层次结构中的功能,比如日志记录、性能监控、安全性验证等。 #### 核心思想 AOP 的核心思想在于将这些横切关注点从业务逻辑中分离出来,从而使代码更加清晰和易于维护。通过这种方式,开发者能够专注于业务逻辑本身,而不需要在每一处涉及的地方重复编写相同的代码片段[^2]。 #### 关键术语 以下是 AOP 中的一些重要术语及其定义: 1. **Aspect(切面)** 切面是一个模块化的单元,包含了多个切入点以及相应的通知行为。它代表了一个特定的关注点,例如日志记录或者事务管理[^3]。 2. **Join Point(连接点)** 连接点指的是程序执行过程中的某些具体位置,在这些地方可以插入额外的行为。通常情况下,连接点会对应于方法调用或者是字段访问操作[^1]。 3. **Advice(通知/建议)** 通知是在某个特定的时间节点所采取的动作。根据触发时机的不同,它可以分为多种类型: - Before Advice:在目标方法之前运行的通知。 - After Returning Advice:仅当目标方法成功完成之后才被执行的通知。 - After Throwing Advice:只有当目标方法抛出了异常时才会激活的通知。 - Around Advice:环绕整个目标方法执行周期的通知,允许自定义控制流程并决定是否继续调用原方法[^3]。 4. **Pointcut(切入点)** 定义了一组匹配条件用来挑选合适的 Join Points 并应用对应的 Advices 上去。简单来说就是一个表达式指定了哪些方法应该受到影响[^2]。 5. **Weaving(织入)** 将 Aspects 合并至最终的应用组件的过程称为 Weaving 或者编织。这可以在编译期、加载期间或是运行时期发生[^4]。 #### Spring框架下的实现机制 Spring 提供了自己的 AOP 支持,并基于代理模式实现了动态代理的功能。对于接口类型的 Bean,默认采用 JDK 动态代理;而对于没有接口的情况,则使用 CGLIB 库创建子类形式的代理实例[^3]。 下面展示如何配置一个简单的 `@Around` 类型的通知示例: ```java import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.Around; import org.aspectj.lang.annotation.Aspect; @Aspect public class LoggingAspect { @Around("execution(* com.example.service..*(..))") public Object logExecutionTime(ProceedingJoinPoint joinPoint) throws Throwable { long start = System.currentTimeMillis(); try { return joinPoint.proceed(); // 调用实际的方法 } finally { long executionTime = System.currentTimeMillis() - start; System.out.println(joinPoint.getSignature() + " executed in " + executionTime + "ms"); } } } ``` 此代码段展示了如何围绕服务层的所有公共方法添加时间测量逻辑而不改变原有函数签名。 ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值