当谈论AOP(面向切面编程)时,我们在软件设计中引入了一种编程范式,用于解决关注点分离的问题。关注点分离是将一个应用程序的不同关注点(例如日志记录、事务管理、安全性等)从业务逻辑中分离出来,以便提高代码的模块化和可维护性。
以下是AOP的主要概念和特性:
切面是一个包含横切关注点逻辑的模块。它定义了在何处(切点)以及何时(通知)执行关注点的逻辑。在实际场景中,我们可以创建一个日志记录切面来记录方法的调用和执行时间。
@Aspect
@Component
public class LoggingAspect {
@Around("execution(* com.example.myapp.service.*.*(..))")
public Object logExecutionTime(ProceedingJoinPoint joinPoint) throws Throwable {
long startTime = System.currentTimeMillis();
Object result = joinPoint.proceed();
long endTime = System.currentTimeMillis();
String methodName = joinPoint.getSignature().getName();
System.out.println("Method " + methodName + " executed in " + (endTime - startTime) + "ms");
return result;
}
}
在这个示例中,@Aspect
注解标记了该类为切面,@Around
通知定义了在哪个切点上执行日志记录逻辑。切点表达式execution(* com.example.myapp.service.*.*(..))
匹配了com.example.myapp.service
包下的所有方法。
切点定义了哪些方法将会被影响,即在哪些方法上应用切面的通知。在实际应用中,我们可以定义一个切点,用于匹配所有UserService
类的方法。
在这个示例中,我们定义了三个不同的切点并应用了不同类型的通知。serviceMethods()切点匹配com.example.myapp.service包中的所有方法。afterRepositoryMethods()切点匹配com.example.myapp.repository包中的所有方法。loggableMethods()切点匹配带有@Loggable注解的方法。
@Aspect
@Component
public class LoggingAspect {
@Pointcut("execution(* com.example.myapp.service.*.*(..))")
public void serviceMethods() {}
@Before("serviceMethods()")
public void beforeAdvice() {
System.out.println("Before method execution");
}
@After("execution(* com.example.myapp.repository.*.*(..))")
public void afterRepositoryMethods() {
System.out.println("After repository method execution");
}
@Around("@annotation(com.example.myapp.annotation.Loggable)")
public Object loggableMethods(ProceedingJoinPoint joinPoint) throws Throwable {
System.out.println("Before method execution");
Object result = joinPoint.proceed();
System.out.println("After method execution");
return result;
}
}
在这个示例中,@Pointcut
注解定义了一个切点,其切点表达式指定了匹配UserService
类的所有方法。你可以在通知中引用这个切点来应用切面逻辑。
通知是切面在特定切点处执行的代码。在实际场景中,我们可以创建一个记录日志的切面,其中包括在目标方法执行前后记录日志的通知。
@Aspect
@Component
public class LoggingAspect {
@Before("userServiceMethods()")
public void beforeAdvice() {
System.out.println("Before method execution");
}
@After("userServiceMethods()")
public void afterAdvice() {
System.out.println("After method execution");
}
}
在这个示例中,@Before
和@After
注解定义了通知,它们分别在切点匹配的方法执行前和执行后执行。userServiceMethods()
是之前定义的切点。
引入是一种增强方式,允许在现有的类中添加新的方法和属性。在实际应用中,我们可以为现有的类引入一个新的接口,以添加新的功能,而无需修改现有代码。
@Aspect
@Component
public class IntroductionAspect {
@DeclareParents(value = "com.example.myapp.service.*+", defaultImpl = AuditableServiceImpl.class)
private AuditableService auditableService;
}
在这个示例中,我们通过@DeclareParents
注解将AuditableService
接口引入到com.example.myapp.service
包下的所有类中。这使得这些类都具备了AuditableService
接口的功能。
AOP的引入(Introduction)和依赖注入
依赖注入(Dependency Injection):
AOP的引入(Introduction):
尽管AOP的引入和依赖注入都涉及在类中添加新功能,但它们的重点和用途是不同的。依赖注入更关注解耦和和可测试性,而AOP的引入更关注在不破坏现有代码的情况下添加新功能。
织入是将切面与应用程序代码结合的过程。在实际应用中,织入将切面的通知插入到切点处,从而实现关注点分离。
@Aspect
@Component
public class LoggingAspect {
@Around("userServiceMethods()")
public Object logExecutionTime(ProceedingJoinPoint joinPoint) throws Throwable {
System.out.println("Before method execution");
Object result = joinPoint.proceed();
System.out.println("After method execution");
return result;
}
}
连接点是程序执行的某个位置,如方法的调用或异常的抛出。
示例:
@AfterReturning(pointcut = "serviceMethods()", returning = "result")
public void logAfterReturning(JoinPoint joinPoint, Object result) {
// ...
}
在这个示例中,logAfterReturning
方法的joinPoint
参数是一个连接点。
在这个示例中,@Around
注解定义了织入的通知。它在目标方法执行前后执行,并将日志记录的逻辑织入到目标方法中。
目标对象是被一个或多个切面通知的对象。
示例:
@Service
public class ExampleService {
public String doSomething(String name) {
return "Hello, " + name;
}
}
在这个示例中,ExampleService
是一个目标对象,它被LoggingAspect
切面通知。
AopProxyUtils
类是Spring AOP框架的一个工具类,它提供了一些静态方法,用于处理代理对象和目标对象。
getSingletonTarget(Object candidate):
这个方法用于获取单例bean的目标对象。
示例:
ExampleService targetObject = (ExampleService) AopProxyUtils.getSingletonTarget(exampleService);
在这个示例中,exampleService
是ExampleService
的代理对象。我们使用AopProxyUtils.getSingletonTarget
方法获取exampleService
的目标对象。
注意:这个方法只适用于单例bean。如果bean的作用域不是单例,这个方法将返回null
。
getTargetClass(Object candidate):
这个方法用于获取代理对象的目标类。
示例:
Class<?> targetClass = AopProxyUtils.getTargetClass(exampleService);
在这个示例中,exampleService
是ExampleService
的代理对象。我们使用AopProxyUtils.getTargetClass
方法获取exampleService
的目标类。
注意:这个方法返回的是目标类,而不是目标对象。
ultimateTargetClass(Object candidate):
这个方法用于获取代理对象的最终目标类。
示例:
Class<?> ultimateTargetClass = AopProxyUtils.ultimateTargetClass(exampleService);
在这个示例中,exampleService
是ExampleService
的代理对象。我们使用AopProxyUtils.ultimateTargetClass
方法获取exampleService
的最终目标类。
注意:这个方法返回的是最终目标类,而不是目标对象。如果代理对象有多层代理,这个方法将返回最终的目标类。
这些是AopProxyUtils
类的一些常用方法。这个类还有一些其他方法,但它们通常不需要在应用程序代码中直接使用。
注意:通常我们不需要直接访问目标对象。代理对象会将调用转发到目标对象,并在调用之前或之后执行通知。所以,通常我们应该使用代理对象,而不是目标对象。直接访问目标对象会绕过代理,这意味着切面的通知将不会被执行。
这个类还包含一些其他的方法,但是它们主要用于内部使用,通常不需要在应用程序代码中直接使用。例如,AopProxyUtils.completeProxiedInterfaces
方法用于确定给定的代理配置的完整代理接口集,包括从目标类继承的接口。这个方法通常用于在创建代理对象时确定代理接口。
AOP在实际开发中的应用场景包括:
当涉及到AOP的实际应用场景时,让我们从五个不同的方面来示例化讲解:
示例:在一个Web应用中,记录每个请求的处理时间和请求参数。
@Aspect
@Component
public class LoggingAspect {
@Around("execution(* com.example.myapp.controller.*.*(..))")
public Object logExecutionTime(ProceedingJoinPoint joinPoint) throws Throwable {
long startTime = System.currentTimeMillis();
Object result = joinPoint.proceed();
long endTime = System.currentTimeMillis();
String methodName = joinPoint.getSignature().getName();
System.out.println("Method " + methodName + " executed in " + (endTime - startTime) + "ms");
return result;
}
}
示例:确保在调用服务方法时,事务在适当的时候启动、提交或回滚。
@Aspect
@Component
public class TransactionAspect {
@Around("@annotation(org.springframework.transaction.annotation.Transactional)")
public Object manageTransaction(ProceedingJoinPoint joinPoint) throws Throwable {
TransactionStatus transactionStatus = transactionManager.getTransaction(new DefaultTransactionDefinition());
try {
Object result = joinPoint.proceed();
transactionManager.commit(transactionStatus);
return result;
} catch (Exception ex) {
transactionManager.rollback(transactionStatus);
throw ex;
}
}
}
示例:在敏感操作前,检查用户是否有足够的权限执行操作。
@Aspect
@Component
public class SecurityAspect {
@Before("@annotation(com.example.myapp.annotation.RequiresAdminRole)")
public void checkAdminRole() {
if (!currentUserHasAdminRole()) {
throw new SecurityException("Admin role required");
}
}
}
示例:在关键方法中添加性能监控和优化逻辑,如数据库查询。
@Aspect
@Component
public class PerformanceAspect {
@Around("execution(* com.example.myapp.repository.*.*(..))")
public Object measureQueryPerformance(ProceedingJoinPoint joinPoint) throws Throwable {
long startTime = System.currentTimeMillis();
Object result = joinPoint.proceed();
long endTime = System.currentTimeMillis();
if (endTime - startTime > 1000) {
System.out.println("Query execution took more than 1 second");
}
return result;
}
}
示例:在方法中添加缓存逻辑,提高应用程序的响应速度。
@Aspect
@Component
public class CachingAspect {
private Map<String, Object> cache = new HashMap<>();
@Around("execution(* com.example.myapp.service.*.*(..))")
public Object applyCaching(ProceedingJoinPoint joinPoint) throws Throwable {
String methodName = joinPoint.getSignature().getName();
if (cache.containsKey(methodName)) {
return cache.get(methodName);
} else {
Object result = joinPoint.proceed();
cache.put(methodName, result);
return result;
}
}
}
在Spring Boot应用中使用AOP,你需要导入以下核心依赖以支持AOP功能:
在Maven项目中,你可以将这些依赖添加到pom.xml
文件中:
<dependencies>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-aopartifactId>
dependency>
dependencies>
当涉及到在Spring Boot应用程序中启用AOP时,你可以将@EnableAspectJAutoProxy
注解放置在两个位置:配置类上或入口函数所在的主类上。以下是这两种方式的合并输出:
@EnableAspectJAutoProxy
注解放置在配置类上:import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
@Configuration
@EnableAspectJAutoProxy
public class AppConfig {
// 配置类的其他内容
}
@EnableAspectJAutoProxy
注解放置在入口函数所在的主类上:import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
@SpringBootApplication
@EnableAspectJAutoProxy
public class MyAppApplication {
public static void main(String[] args) {
SpringApplication.run(MyAppApplication.class, args);
}
}
无论你选择哪种方式,它们都会启用AspectJ自动代理,使AOP切面能够正常应用于Spring Boot应用程序中的Bean。根据你的偏好和项目结构,选择合适的位置来放置@EnableAspectJAutoProxy
注解。
切点表达式(Pointcut Expression)是AOP中一个重要的概念,用于指定在哪些方法上应用通知。切点表达式使用AspectJ的语法,它允许你定义一组匹配方法的规则。这些规则可以基于方法的名称、参数、返回类型等来进行匹配。以下是切点表达式的详细说明和示例:
切点表达式的基本语法是使用关键字execution
,后面跟着方法的返回类型、类名、方法名和参数列表。通配符可以用来匹配不同的部分。
execution([可见性] 返回类型 [类全名.]方法名(参数列表) [异常模式])
其中,方括号内的部分是可选的,具体的匹配规则如下:
可见性
:方法的可见性,如public
、private
等。返回类型
:方法的返回类型,使用*
通配符表示任意类型。类全名
:类的全名,使用*
通配符表示任意包名,省略则表示当前包下的所有类。方法名
:方法的名称,使用*
通配符表示任意方法名。参数列表
:方法的参数列表,使用..
表示任意数量和类型的参数。异常模式
:方法可能抛出的异常。假设我们有以下服务类:
package com.example.myapp.service;
@Service
public class UserService {
public void createUser(String username) {
System.out.println("User created: " + username);
}
public void deleteUser(String username) {
System.out.println("User deleted: " + username);
}
}
我们来定义一些切点表达式示例:
UserService
类的方法:@Pointcut("execution(* com.example.myapp.service.UserService.*(..))")
public void userServiceMethods() {}
@Pointcut("execution(public * *(..))")
public void publicMethods() {}
create
开头的方法:@Pointcut("execution(* create*(..))")
public void createMethods() {}
void
的方法:@Pointcut("execution(void *(..))")
public void voidReturnMethods() {}
String
类型参数的方法:@Pointcut("execution(* *(String))")
public void stringParameterMethods() {}
int
类型,第二个参数是任意类型:@Pointcut("execution(* *(int, ..))")
public void intAndAnyParameterMethods() {}
示例:定义一个用于日志记录的切面类。
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;
@Aspect
@Component
public class LoggingAspect {
@Before("execution(* com.example.myapp.service.*.*(..))")
public void beforeAdvice() {
System.out.println("Before method execution");
}
@After("execution(* com.example.myapp.service.*.*(..))")
public void afterAdvice() {
System.out.println("After method execution");
}
}
@Pointcut
注解。@Pointcut
注解的方法体中。示例:定义一个切点来匹配服务类中的所有方法。
@Pointcut("execution(* com.example.myapp.service.*.*(..))")
public void serviceMethods() {}
示例:在调用服务方法之前打印日志。
@Before("serviceMethods()")
public void beforeAdvice() {
System.out.println("Before method execution");
}
示例:在调用服务方法之后打印日志。
@After("serviceMethods()")
public void afterAdvice() {
System.out.println("After method execution");
}
示例:在调用服务方法成功执行后打印日志。
@AfterReturning(pointcut = "serviceMethods()", returning = "result")
public void afterReturningAdvice(Object result) {
System.out.println("Method returned: " + result);
}
示例:在调用服务方法抛出异常后打印日志。
@AfterThrowing(pointcut = "serviceMethods()", throwing = "exception")
public void afterThrowingAdvice(Exception exception) {
System.out.println("Exception thrown: " + exception.getMessage());
}
ProceedingJoinPoint
参数,通过调用其proceed()
方法来执行目标方法。示例:在调用服务方法前后记录执行时间。
@Around("serviceMethods()")
public Object aroundAdvice(ProceedingJoinPoint joinPoint) throws Throwable {
long startTime = System.currentTimeMillis();
Object result = joinPoint.proceed();
long endTime = System.currentTimeMillis();
System.out.println("Method execution time: " + (endTime - startTime) + "ms");
return result;
}
示例:定义切面的执行顺序。
@Aspect
@Component
@Order(1)
public class FirstAspect {
// ...
}
@Aspect
@Component
@Order(2)
public class SecondAspect {
// ...
}
当谈论AOP中的切面注解时,除了基于切点表达式匹配方法外,还可以使用注解作为切点。这种方式允许你在特定注解被使用时触发通知,从而实现一种基于注解的AOP。以下是涉及切面注解和监听注解的解释和示例:
假设我们有一个使用了自定义注解@Loggable
的服务类:
package com.example.myapp.service;
import org.springframework.stereotype.Service;
@Service
public class UserService {
@Loggable
public void createUser(String username) {
System.out.println("User created: " + username);
}
@Loggable
public void deleteUser(String username) {
System.out.println("User deleted: " + username);
}
}
我们可以创建一个切面来监听使用了@Loggable
注解的方法,并在这些方法执行前后记录日志:
package com.example.myapp.aspect;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;
@Aspect
@Component
public class LoggingAspect {
@Around("@annotation(com.example.myapp.annotation.Loggable)")
public Object logAround(ProceedingJoinPoint joinPoint) throws Throwable {
System.out.println("Before method execution");
Object result = joinPoint.proceed();
System.out.println("After method execution");
return result;
}
}
在上述示例中,我们使用了@Around
注解并指定切点表达式为@annotation(com.example.myapp.annotation.Loggable)
,这将匹配使用了@Loggable
注解的方法。在切面的logAround
方法中,我们在方法执行前后记录了日志。