Spring AOP原理:动态代理与AspectJ对比

Spring AOP原理:动态代理与AspectJ对比

Spring框架是Java后端开发的核心技术栈,掌握其原理和最佳实践至关重要。

一、Spring核心特性

AOP是Spring的核心功能,理解动态代理和AspectJ的实现原理对解耦和横切逻辑至关重要

二、IOC容器

2.1 Bean的生命周期

Spring Bean 生命周期流程:

实例化
  ↓
属性赋值 (populateBean)
  ↓
BeanNameAware (setBeanName)
  ↓
BeanFactoryAware (setBeanFactory)
  ↓
ApplicationContextAware (setApplicationContext)
  ↓
BeanPostProcessor前置 (postProcessBeforeInitialization)
  ↓
InitializingBean (afterPropertiesSet)
  ↓
init-method (自定义初始化方法)
  ↓
BeanPostProcessor后置 (postProcessAfterInitialization)
  ↓
使用Bean (ready for use)
  ↓
DisposableBean (destroy)
  ↓
destroy-method (自定义销毁方法)

说明:
- 蓝色节点:Spring 提供的扩展接口
- 绿色节点:用户自定义的方法

2.2 依赖注入方式

// 1. 构造器注入(推荐)
@Component
public class UserService {
    private final UserRepository userRepository;

    @Autowired
    public UserService(UserRepository userRepository) {
        this.userRepository = userRepository;
    }
}

// 2. Setter注入
@Component
public class UserService {
    private UserRepository userRepository;

    @Autowired
    public void setUserRepository(UserRepository userRepository) {
        this.userRepository = userRepository;
    }
}

// 3. 字段注入(不推荐)
@Component
public class UserService {
    @Autowired
    private UserRepository userRepository;
}

三、AOP面向切面编程

3.1 AOP核心概念

  • 切面(Aspect):横切关注点的模块化
  • 连接点(JoinPoint):程序执行的某个位置
  • 切点(Pointcut):匹配连接点的表达式
  • 通知(Advice):在切点上执行的动作
  • 目标对象(Target):被代理的对象

3.2 通知类型

@Aspect
@Component
public class LoggingAspect {

    // 前置通知
    @Before("execution(* com.example.service.*.*(..))")
    public void beforeMethod(JoinPoint joinPoint) {
        System.out.println("Before: " + joinPoint.getSignature());
    }

    // 后置通知
    @After("execution(* com.example.service.*.*(..))")
    public void afterMethod(JoinPoint joinPoint) {
        System.out.println("After: " + joinPoint.getSignature());
    }

    // 返回通知
    @AfterReturning(pointcut = "execution(* com.example.service.*.*(..))",
                     returning = "result")
    public void afterReturning(JoinPoint joinPoint, Object result) {
        System.out.println("AfterReturning: " + result);
    }

    // 异常通知
    @AfterThrowing(pointcut = "execution(* com.example.service.*.*(..))",
                   throwing = "ex")
    public void afterThrowing(JoinPoint joinPoint, Exception ex) {
        System.out.println("AfterThrowing: " + ex.getMessage());
    }

    // 环绕通知
    @Around("execution(* com.example.service.*.*(..))")
    public Object aroundMethod(ProceedingJoinPoint joinPoint) throws Throwable {
        System.out.println("Around Before");
        Object result = joinPoint.proceed();
        System.out.println("Around After");
        return result;
    }
}

四、Spring事务管理

4.1 事务注解

@Service
public class OrderService {

    // 声明式事务
    @Transactional(
        propagation = Propagation.REQUIRED,  // 传播行为
        isolation = Isolation.READ_COMMITTED,  // 隔离级别
        timeout = 30,  // 超时时间
        readOnly = false,  // 只读
        rollbackFor = Exception.class  // 回滚异常
    )
    public void createOrder(Order order) {
        // 业务逻辑
    }
}

4.2 事务传播行为

传播行为 说明
REQUIRED 默认,如果存在事务则加入,否则新建
REQUIRES_NEW 新建事务,挂起当前事务
SUPPORTS 如果存在事务则加入,否则非事务执行
NOT_SUPPORTED 非事务执行,挂起当前事务
MANDATORY 必须在事务中,否则抛异常
NEVER 必须非事务执行,否则抛异常
NESTED 嵌套事务,回滚点保存

五、Spring Boot自动配置

5.1 自动配置原理

// @EnableAutoConfiguration 注解
@AutoConfigurationPackage
@Import(AutoConfigurationImportSelector.class)
public @interface EnableAutoConfiguration {
    String[] excludeName() default {};
}

// 自动配置类示例
@Configuration(proxyBeanMethods = false)
@ConditionalOnClass(DataSource.class)
@ConditionalOnProperty(prefix = "spring.datasource", name = "url")
@EnableConfigurationProperties(DataSourceProperties.class)
public class DataSourceAutoConfiguration {
    // 配置逻辑
}

5.2 常用注解

// 条件注解
@ConditionalOnClass        // 类路径下存在指定类
@ConditionalOnMissingClass  // 类路径下不存在指定类
@ConditionalOnBean         // 容器中存在指定Bean
@ConditionalOnMissingBean   // 容器中不存在指定Bean
@ConditionalOnProperty      // 配置属性满足条件
@ConditionalOnWebApplication  // Web环境

六、常见面试题

Q1: Spring中循环依赖如何解决?

答案:
Spring通过三级缓存解决循环依赖:

// 三级缓存
1. singletonObjects          // 一级缓存:完整的Bean
2. earlySingletonObjects     // 二级缓存:提前暴露的Bean
3. singletonFactories        // 三级缓存:Bean的工厂

// 解决流程
1. A实例化后,放入三级缓存
2. A注入B,B需要A
3. B从三级缓存获取A的工厂,创建A的代理对象
4. B注入A的代理对象,B完成初始化
5. A继续初始化,注入B,完成初始化

Q2: @Autowired和@Resource的区别?

答案:

特性 @Autowired @Resource
来源 Spring Java JSR-250
匹配方式 按类型,其次按名称 按名称,其次按类型
Required属性 支持 不支持
作用位置 构造器、方法、字段 方法、字段

Q3: 事务失效的场景有哪些?

答案:
1. 方法不是public
2. 方法自调用(this调用)
3. 异常被捕获未抛出
4. 异常类型不匹配(默认RuntimeException)
5. 数据库引擎不支持事务
6. 未被Spring管理(无@Component等注解)

七、最佳实践

代码规范:
- 优先使用构造器注入
- 合理使用AOP,避免过度设计
- 事务粒度要小,避免大事务

性能优化:
- 使用循环依赖缓存
- 合理使用懒加载(@Lazy)
- 控制Bean的生命周期

八、总结

Spring框架博大精深,持续学习:

核心要点
- 掌握IOC和AOP原理
- 理解事务传播机制
- 熟悉自动配置原理

进阶方向
- 阅读Spring源码
- 学习Spring Cloud微服务
- 实践项目应用

推荐资源
- 《Spring实战》
- Spring官方文档
- Spring源码分析


posted @ 2026-02-21 23:00  寒人病酒  阅读(0)  评论(0)    收藏  举报