一、什么是切面
AOP,即面向切面编程,目的是为了统一处理某些内容,例如日志处理。
如果我们在每一地方都分别做处理,慢慢地代码就会变得很难维护。
以日志处理为例,切面就可以做到在每个操作执行完成后进行日志插入,而不用在每个操作的代码里都添加插入日志的方法。
这里介绍通过spring注解方式实现的切面。
二、切面相关注解
注解 | 说明 |
---|---|
@Aspect | 将一个 java 类定义为切面类 |
@Pointcut | 定义一个切入点,可以指定切入点的作用范围 |
@Before | 在切入点开始前执行 |
@After | 在切入点结束后执行 |
@AfterReturning | 在切入点 return 后执行 |
@AfterThrowing | 处理切入点操作异常 |
@Around | 在切入点前后执行 |
三、示例代码
/***
* @author cyd
* @date 2023/7/31 10:04
* @description <测试切面----打印>
**/
@Aspect
@Component
public class printTestAspect {
// 定义一个切入点
@Pointcut("execution(* com.pw.service.*.*(..))")//切入点为service下的所有方法
public void print() {
System.out.println("print操作");
}
// 前置通知
@Before(value = "print()")//代表使用print()这个切入点,print()中的内容不执行
public void before(JoinPoint jp) {
//获取方法名
String name = jp.getSignature().getName();
System.out.println("[" + name + "]方法开始执行...");
}
// 后置通知
@After(value = "print()")
public void after(JoinPoint jp) {
String name = jp.getSignature().getName();
System.out.println("[" + name + "]方法执行结束...");
}
// 返回通知
@AfterReturning(value = "print()", returning = "result")
public void afterReturning(JoinPoint jp, Object result) {
String name = jp.getSignature().getName();
System.out.println("[" + name + "]方法返回值为:" + result);
}
// 异常通知
@AfterThrowing(value = "print()", throwing = "e")
public void afterThrowing(JoinPoint jp, Exception e) {
String name = jp.getSignature().getName();
System.out.println("[" + name + "]方法出现异常:" + e.getMessage());
}
// 环绕通知
@Around("print()")
public Object around(ProceedingJoinPoint pjp) throws Throwable {
String name = pjp.getSignature().getName();
// 计算方法执行时间
long start = System.currentTimeMillis();
Object result = pjp.proceed();
long end = System.currentTimeMillis();
System.out.println("[" + name + "]方法执行时间为:" + (end - start) + " ms");
return result;
}
}
四、运行结果
随便执行一个方法
[getById]方法开始执行...
Creating a new SqlSession
SqlSession [org.apache.ibatis.session.defaults.DefaultSqlSession@7d4e45a6] was not registered for synchronization because synchronization is not active
2024-01-10 10:06:57.555 INFO 29444 --- [nio-8080-exec-2] com.zaxxer.hikari.HikariDataSource : HikariPool-1 - Starting...
2024-01-10 10:06:57.670 INFO 29444 --- [nio-8080-exec-2] com.zaxxer.hikari.HikariDataSource : HikariPool-1 - Start completed.
JDBC Connection [HikariProxyConnection@1605304306 wrapping com.mysql.cj.jdbc.ConnectionImpl@33d4323a] will not be managed by Spring
==> Preparing: SELECT user_id,nick_name,account,password,email,phone,avatar,introduction,sex,create_by,create_time,update_by,update_time FROM user WHERE user_id=?
==> Parameters: 1654052760859000833(String)
<== Columns: user_id, nick_name, account, password, email, phone, avatar, introduction, sex, create_by, create_time, update_by, update_time
<== Row: 1654052760859000833, tom, null, null, null, null, null, null, null, null, null, null, null
<== Total: 1
Closing non transactional SqlSession [org.apache.ibatis.session.defaults.DefaultSqlSession@7d4e45a6]
[getById]方法返回值为:User(userId=1654052760859000833, nickName=tom, account=null, password=null, email=null, phone=null, avatar=null, introduction=null, sex=null)
[getById]方法执行结束...
[getById]方法执行时间为:193 ms