您好,登录后才能下订单哦!
# Spring如何实现AOP
## 目录
1. [AOP核心概念解析](#1-aop核心概念解析)
2. [Spring AOP的实现原理](#2-spring-aop的实现原理)
3. [代理模式深度剖析](#3-代理模式深度剖析)
4. [动态字节码生成技术](#4-动态字节码生成技术)
5. [Spring AOP配置全解](#5-spring-aop配置全解)
6. [性能优化与最佳实践](#6-性能优化与最佳实践)
7. [复杂场景解决方案](#7-复杂场景解决方案)
8. [与其他技术的对比](#8-与其他技术的对比)
9. [未来发展趋势](#9-未来发展趋势)
## 1. AOP核心概念解析
### 1.1 什么是AOP
面向切面编程(Aspect-Oriented Programming)是一种通过预编译方式和运行期动态代理实现程序功能统一维护的技术。与OOP的纵向继承不同,AOP采用横向抽取机制,将分散在各个方法中的重复代码提取出来,在程序编译或运行时再植入到需要的地方。
**典型应用场景**:
- 日志记录
- 事务管理
- 权限控制
- 性能监控
- 异常处理
### 1.2 AOP核心术语体系
| 术语 | 说明 |
|---------------|----------------------------------------------------------------------|
| Aspect(切面) | 封装横切逻辑的模块,包含Pointcut和Advice |
| Joinpoint | 程序执行过程中的特定点(如方法调用、异常抛出) |
| Pointcut | 匹配Joinpoint的谓词,定义Advice的执行时机 |
| Advice | 切面在特定Joinpoint执行的动作(前/后/环绕/异常/最终) |
| Weaving | 将切面应用到目标对象创建新代理对象的过程(编译期/类加载期/运行期) |
### 1.3 Spring AOP与AspectJ对比
```java
// Spring AOP示例
@Aspect
@Component
public class LoggingAspect {
@Before("execution(* com.example.service.*.*(..))")
public void logBefore(JoinPoint joinPoint) {
System.out.println("方法执行前: " + joinPoint.getSignature().getName());
}
}
// AspectJ示例(需要编译器支持)
public aspect LoggingAspect {
before(): call(* com.example.service.*.*(..)) {
System.out.println("方法调用前: " + thisJoinPoint.getSignature().getName());
}
}
关键差异: - Spring AOP基于动态代理,AspectJ使用字节码增强 - Spring AOP仅支持方法级拦截,AspectJ支持字段/构造器拦截 - Spring AOP运行时织入,AspectJ支持编译时/后织入
graph TD
A[ProxyFactory] --> B[AdvisedSupport]
B --> C[TargetSource]
B --> D[AdvisorChain]
D --> E[PointcutAdvisor]
E --> F[Pointcut]
E --> G[Advice]
A --> H[ProxyCreatorSupport]
H --> I[JdkDynamicAopProxy]
H --> J[CglibAopProxy]
public class JdkDynamicProxy implements InvocationHandler {
private final Object target;
private final MethodInterceptor interceptor;
public static Object createProxy(Object target, MethodInterceptor interceptor) {
return Proxy.newProxyInstance(
target.getClass().getClassLoader(),
target.getClass().getInterfaces(),
new JdkDynamicProxy(target, interceptor));
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
MethodInvocation invocation = new ReflectiveMethodInvocation(
target, method, args);
return interceptor.invoke(invocation);
}
}
实现要点: - 基于java.lang.reflect.Proxy构建 - 要求目标类必须实现接口 - 通过InvocationHandler统一处理调用
public class CglibProxyCreator implements MethodInterceptor {
public Object createProxy(Class<?> targetClass) {
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(targetClass);
enhancer.setCallback(this);
return enhancer.create();
}
@Override
public Object intercept(Object obj, Method method, Object[] args,
MethodProxy proxy) throws Throwable {
System.out.println("Before method: " + method.getName());
Object result = proxy.invokeSuper(obj, args);
System.out.println("After method: " + method.getName());
return result;
}
}
性能对比:
维度 | JDK动态代理 | CGLIB |
---|---|---|
创建速度 | 快(缓存机制) | 慢(生成字节码) |
执行性能 | 反射调用略慢 | 直接调用更快 |
目标类要求 | 必须实现接口 | 无要求 |
方法拦截范围 | 仅接口方法 | 所有非final方法 |
ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_MAXS);
MethodVisitor mv = cw.visitMethod(
ACC_PUBLIC, "sayHello", "()V", null, null);
mv.visitCode();
mv.visitFieldInsn(GETSTATIC, "java/lang/System", "out", "Ljava/io/PrintStream;");
mv.visitLdcInsn("Hello World!");
mv.visitMethodInsn(INVOKEVIRTUAL, "java/io/PrintStream", "println", "(Ljava/lang/String;)V", false);
mv.visitInsn(RETURN);
mv.visitMaxs(0, 0);
mv.visitEnd();
@Configuration
@EnableAspectJAutoProxy(proxyTargetClass = true)
public class AppConfig {
@Bean
public LoggingAspect loggingAspect() {
return new LoggingAspect();
}
}
@Aspect
public class LoggingAspect {
@Pointcut("execution(* com..service.*.*(..))")
private void serviceLayer() {}
@Around("serviceLayer()")
public Object logExecutionTime(ProceedingJoinPoint joinPoint) throws Throwable {
long start = System.currentTimeMillis();
Object result = joinPoint.proceed();
long duration = System.currentTimeMillis() - start;
System.out.println(joinPoint.getSignature() + " executed in " + duration + "ms");
return result;
}
}
<aop:config proxy-target-class="true">
<aop:aspect id="logAspect" ref="loggingAspect">
<aop:pointcut id="servicePointcut"
expression="execution(* com.example.service.*.*(..))"/>
<aop:around method="logExecutionTime" pointcut-ref="servicePointcut"/>
</aop:aspect>
</aop:config>
<bean id="loggingAspect" class="com.example.aspect.LoggingAspect"/>
spring.aop.proxy-target-class=true # 强制使用CGLIB
// 推荐 - 精确限定包路径 @Pointcut(“execution(* com.example.service.UserService.*(..))”)
2. **代理选择策略**:
- 接口优先使用JDK动态代理
- 需要代理非接口方法时选择CGLIB
3. **缓存配置**:
```java
@Aspect
public class CachedAspect {
private ConcurrentMap<Object, Object> cache = new ConcurrentHashMap<>();
@Around("@annotation(cacheable)")
public Object cacheResult(ProceedingJoinPoint pjp, Cacheable cacheable) throws Throwable {
String key = createKey(pjp);
return cache.computeIfAbsent(key, k -> pjp.proceed());
}
}
自调用问题:
public class UserService {
public void updateUser(User user) {
// 自调用不会触发AOP
this.validateUser(user);
}
@Transactional
public void validateUser(User user) {
// 事务注解失效
}
}
异常处理误区:
@AfterThrowing(pointcut="serviceLayer()", throwing="ex")
public void handleException(JoinPoint jp, Exception ex) {
// 捕获异常后默认不会继续传播
if(ex instanceof BusinessException) {
throw new CustomException("转换异常类型", ex);
}
}
@Aspect
public class MultiDataSourceAspect {
@Around("@annotation(targetDataSource)")
public Object switchDataSource(ProceedingJoinPoint pjp,
TargetDataSource targetDataSource) throws Throwable {
String oldKey = DynamicDataSourceHolder.getDataSourceKey();
try {
DynamicDataSourceHolder.setDataSourceKey(targetDataSource.value());
return pjp.proceed();
} finally {
DynamicDataSourceHolder.setDataSourceKey(oldKey);
}
}
}
@Aspect
public class DistributedLockAspect {
@Autowired
private RedissonClient redissonClient;
@Around("@annotation(redisLock)")
public Object doWithLock(ProceedingJoinPoint pjp, RedisLock redisLock) throws Throwable {
RLock lock = redissonClient.getLock(buildLockKey(pjp, redisLock));
try {
if(lock.tryLock(redisLock.waitTime(), redisLock.leaseTime(), redisLock.unit())) {
return pjp.proceed();
}
throw new LockAcquisitionException("获取分布式锁失败");
} finally {
if(lock.isHeldByCurrentThread()) {
lock.unlock();
}
}
}
}
对比维度 | Spring AOP | AspectJ |
---|---|---|
织入方式 | 运行时动态代理 | 编译期/类加载期字节码增强 |
性能开销 | 每次调用有反射开销 | 无运行时开销 |
功能范围 | 仅方法级别 | 字段/构造器/静态代码块 |
学习曲线 | 简单 | 需要掌握特殊语法 |
依赖要求 | 仅需Spring Core | 需要编译器或类加载器支持 |
注:本文为缩减版示例,完整版将包含: - 更多代码示例(约30个) - 详细的性能测试数据 - 完整的UML类图 - 实际项目案例解析 - 各版本特性对比表格 “`
实际完整文章将包含: 1. 深入分析Spring AOP的12个核心类 2. 5种典型业务场景的实现方案 3. 3种性能优化技巧的基准测试 4. 与Spring Boot/Cloud的整合指南 5. 常见问题的15个解决方案
需要展开任何部分请告知,我可提供更详细的内容补充。
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。