Java SpringBoot实现AOP
作者:Tttori 发布时间:2023-05-31 05:49:30
1、AOP基本总结
连接点(
JoinPoint
):连接点是程序运行的某个阶段点,如方法调用、异常抛出等
切入点(
Pointcut
):切入点是
JoinPoint
的集合
是程序中需要注入Advice
的位置的集合,即Advice
在什么条件下才能被触发增强(
Advisor
):增强是切入点
Pointcut
和Advice
的综合体,即在连接点JoinPoint
上执行的行为
通过JDK/CGLIB
代理模式实现AOP切面(
Aspect
):@Aspect通常是一个类的注解,通常与
@Component
搭配使用AOP代理(
AOP Proxy
):AOP使用 * 模式创建对象,从而实现在连接点
JoinPoint
处插入增强
其中JDK只能代理接口,CGLIB基于子类但不能代理final类
2、常用方法
3、增强类型
@Before
:前置增强,在某个JoinPoint
执行前的增强@After
:final增强,不管抛异常还是正常退出都执行的增强@AfterReturning
:后置增强,方法正常退出时执行@AfterThrowing
:异常抛出增强,抛出异常后执行@Around
:环绕增强,包围一个连接点的增强,最强大的一个方式,且常用
4、示例说明
学了一下AOP的使用,写了个@Around
的demo
,将几个查询操作存入数据库作为Log
并且定时清理过期数据
本人的Demo
用的是Dubbo
框架,而AOP的示例写在了Provider
中,大概结构如下:
monitor:
annotation
:注解类aop
:切面的定义及实现impl
:UserAopTask接口的实现类
1)UserLog
实体类
@Data
@ToString
@NoArgsConstructor
@AllArgsConstructor
public class UserLog implements Serializable {
private Integer id;
private String methodName;
private String methodArgs;
private String classFullName;
private String className;
private Date invokeTime;
private Double costTime;
}
2)LogTaskMapper
对应mapper
接口
public interface LogTaskMapper {
/**
* TEST AOP INSERT INFO INTO TABLE
* @param userLog
*/
void insertUserLog(UserLog userLog);
/**
* DELETE LOGS IN TABLE LAST x MINUTES
* @param minutes
*/
void deleteUserLog(int minutes);
}
3)UserAopTask
接口
public interface UserAopTask {
void insertUserLog(UserLog log);
}
4)UserAopTaskImpl
实现类
@Component
public class UserAopTaskImpl implements UserAopTask {
private static final Logger logger = LoggerFactory.getLogger(UserAopTask.class);
@Autowired
private LogTaskMapper logTaskMapper;
private ExecutorService logHandler = Executors.newFixedThreadPool(1);//采用线程池复用一个线程执行
private static final int MINUTES_LOG_RETAIN = 30;//数据库中数据保留时间
@Override
public void insertUserLog(UserLog log) {
logHandler.submit(new logSubmitTask(log));
}
//内部类
class logSubmitTask implements Runnable{
private UserLog userLog;
public logSubmitTask(UserLog userLog){
this.userLog = userLog;
}
@Override
public void run() {
logTaskMapper.insertUserLog(userLog);
}
}
//定时清理任务
@Scheduled(cron = "0 30 * * * *")
public void scheduledDeleteLog(){
logger.info("开始清除[{}]分钟之前的图表查询日志...", MINUTES_LOG_RETAIN);
logTaskMapper.deleteUserLog(-1 * MINUTES_LOG_RETAIN);
}
}
5)TestUserAop
切面类
@Aspect//切面
@Component//Spring容器管理
public class TestUserAop {
private static final Logger logger = LoggerFactory.getLogger(TestUserAop.class);
@Autowired
private UserAopTask userAopTask;
//使用环绕增强,第一参数必须是ProceedingJoinPoint
@Around(value = "@annotation(annotation)")//和注解类参数名保持一致
public Object aroundUserInfo(ProceedingJoinPoint pjp, TestUserAnnotation annotation) throws Throwable{
UserLog userLog = new UserLog();
System.out.println("=====================ANNOTATION BEGIN=====================");
Date date = new Date();
Long methodStart = date.getTime();//timestamp
System.out.println("ANNOTATION 开始耗时统计: "+ date);
userLog.setInvokeTime(date);
Object[] argsObj = pjp.getArgs();
Object res = pjp.proceed(argsObj);//利用反射调用目标方法
Long methodCost = System.currentTimeMillis() - methodStart;
double cost = methodCost/1000d;//timestamp 转换为 seconds
System.out.println("ANNOTATION 调用方法总耗时: "+ String.format("%.3f",cost) +" s");//保留3位小数
System.out.println("ANNOTATION 调用方法: "+annotation.methodName());//目标方法
System.out.println("ANNOTATION 调用方法参数: "+ new Integer((Integer) argsObj[0]));//我的参数就1个或者无参
System.out.println("ANNOTATION 调用类: "+pjp.getSignature().getDeclaringTypeName());//全类名
System.out.println("ANNOTATION 调用类名: "+pjp.getSignature().getDeclaringType().getSimpleName());//类名
System.out.println("ANNOTATION 调用结果: "+ JSON.toJSON(res));
System.out.println("=====================ANNOTATION FINISHED=====================");
userLog.setCostTime(Double.parseDouble(String.format("%.3f",cost)));
userLog.setClassFullName(pjp.getSignature().getDeclaringTypeName());
userLog.setClassName(pjp.getSignature().getDeclaringType().getSimpleName());
userLog.setMethodName(annotation.methodName());
userLog.setMethodArgs(Integer.toString(new Integer((Integer) argsObj[0])));
userAopTask.insertUserLog(userLog);
return res;
}
}
6)TestUserAnnotation
注解类
我在service
层写的AOP demo
,对目标方法使用注解,注解名为注解类名即可,如@TestUserAnnotation
@Retention(RetentionPolicy.RUNTIME)//运行时有效
@Target(ElementType.METHOD)//作用于方法
@Documented
public @interface TestUserAnnotation {
String methodName() default "";//方法名,默认为空字符串
}
7)LogTaskMapper.xml
最后贴个代码,为上面提到的定时任务,用到的是date_add()方法,其中的 "<" 意为 "<"
<delete id="deleteUserLog" parameterType="java.lang.Integer">
delete from invoke_log
where invoke_time < date_add(current_timestamp,interval #{minutes} minute)
</delete>
5、结果展示
演示一下AOP的效果,将@TestUserAnnotation
注解在方法getUserInfo(),
即获取用户信息
Demo
中利用AOP的@Around
环绕增强,实现了统计方法调用运行消耗时间,以及统计调用方法名、类名等信息:
调用方法getUserInfo
后的统计结果:
来源:https://www.cnblogs.com/torima/p/15124164.html
猜你喜欢
- Web UI项目中, 很多 Spring controller 视图函数直接返回 html 页面, 还有一些视图函数是要重定向或转发到其他的
- 目录栈溢出(虚拟机栈和本地方法栈)产生原因解决思路堆溢出产生原因解决思路方法区和运行时常量池溢出产生原因解决思路本机直接内存溢出产生原因解决
- 这篇文章主要介绍了SpringBoot2整合activiti6环境搭建过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定
- 前言我们在 页面切换转场动画,英雄救场更有趣!介绍了 Hero 动画效果,使用 Hero 用于转场能够提供非常不错的体验。既然称之
- 当我们使用swagger,进行接口测试,怕接口不安全,担心暴露。可采用两种方式1.环境权限配置对swagger文档配置只在测试环境可访问,生
- 估计学过Unix开发但是没有细致学习Java的同学们会疑惑了,操作系统里面是没有所谓的守护线程的概念,只有守护进程一说,但是Java语言机制
- 前言小小知识,不值一提,了解了也不能让你提高身价,但是不了解你就是比别人少知道点!事儿就是这么个事儿,直接正题吧! 直接看代码演示优先级当方
- 根据不同系统动态获取换行符和盘分割符1、获取盘分割符File.separator2、获取换行符windows系统为\r\n,Linux系统为
- 本文实例为大家分享了JavaFX实现简单日历效果的具体代码,供大家参考,具体内容如下1.先看效果:2.代码:1)ClockEdge.java
- 导读Lombok:可以让你的POJO代码特别简洁,不止简单在BO/VO/DTO/DO等大量使用,还有设计模式,对象对比等MybatisPlu
- 本文实例讲述了Java数据结构之链表、栈、队列、树的实现方法。分享给大家供大家参考,具体如下:最近无意中翻到一本书,闲来无事写几行代码,实现
- 死锁在多线程的情况下,会出现数据不同步情况, 而为了避免这种情况,之前也说了:界区实现方法有两种,一种是用synchronized,一种是用
- 题目描述Java创建线程的几种方式Java使用Thread类代表线程,所有线程对象都必须是Thread类或者其子类的实例。Java可以用以下
- 这篇文章主要介绍了简单了解Java多态向上转型相关原理,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋
- 前2天有读者问到是否有带分页功能的表格控件,今天分页功能的表格控件详细解析。PaginatedDataTablePaginatedDataT
- 因为公司现在换成了nacos,所以自己写了demo学习一下。结果第一步就走不下去。在使用nacos-config读取nacos配置时。发现b
- 介绍众所周知,AOP(面向切面编程)是Spring框架的特色功能之一。通过设置横切关注点(cross cutting concerns),A
- 合成聚合复用原则合成复用原则又称为组合/聚合复用原则(Composition/Aggregate Reuse Principle, CARP
- 一、使用注解实现自定义映射关系当POJO属性名与数据库列名不一致时,需要自定义实体类和结果集的映射关系,在MyBatis注解开发中,使用 @
- 分析Github 3000个开源项目,粗略统计如下。括号内的数字是使用频率 0-3000. 下面的列表显示不全,完整的请看完整列表。1.ja