Spring AOP 切面编程记录日志和接口执行时间

最近客户现在提出系统访问非常慢,需要优化提升访问速度,在排查了nginx、tomcat内存和服务器负载之后,判断是数据库查询速度慢,进一步排查发现是因为部分视图和表查询特别慢导致了整个系统的响应时间特别长。知道了问题之后,就需要对查询比较慢的接口进行优化,但哪些接口需要优化、哪些不需要呢?只能通过日志里的执行时间来判断,那么如何才能知道每一个接口的执行时间呢?

如果想学习Java工程化、高性能及分布式、深入浅出。微服务、Spring,MyBatis,Netty源码分析的朋友可以加我的Java高级交流:854630135,群里有阿里大牛直播讲解技术,以及Java大型互联网技术的视频免费分享给大家。

对于这个问题,想到了使用动态代理的方式统一记录方法的执行时间并打印日志,这样就能很直观、方便的看到每个接口的执行时间了。

由于使用的是spring框架,对象都是由spring统一管理的,所以最后使用的是 Spring AOP 切面编程来统一记录接口的执行时间,具体代码如下(基于注解的方式):

@Component

@Aspect

public class AopLoggerAspect {

private static final Logger logger = Logger.getLogger(AopLoggerAspect.class);

@Pointcut(execution(public * com.iflytek.credit.platform.*.service.impl.*Impl.*(..)) || execution(public * com.iflytek.credit.platform.*.controller.*Controller.*(..)))

public void pointCut() {

}

@Before(pointCut())

public void boBefore(JoinPoint joinPoint) {

String methodName = joinPoint.getSignature().getName();

logger.info(Method Name : [ + methodName + ] --- AOP before );

}

@After(pointCut())

public void doAfter(JoinPoint joinPoint) {

String methodName = joinPoint.getSignature().getName();

logger.info(Method Name : [ + methodName + ] --- AOP after );

}

@AfterReturning(pointcut = pointCut(),returning = result)

public void afterReturn(JoinPoint joinPoint, Object result) {

String methodName = joinPoint.getSignature().getName();

logger.info(Method Name : [ + methodName + ] --- AOP after return ,and result is : + result.toString());

}

@AfterThrowing(pointcut = pointCut(),throwing = throwable)

public void afterThrowing(JoinPoint joinPoint, Throwable throwable) {

String methodName = joinPoint.getSignature().getName();

logger.info(Method Name : [ + methodName + ] --- AOP after throwing ,and throwable message is : + throwable.getMessage());

}

@Around(pointCut())

public Object around(ProceedingJoinPoint joinPoint) {

String methodName = joinPoint.getSignature().getName();

try {

logger.info(Method Name : [ + methodName + ] --- AOP around start);

long startTimeMillis = System.currentTimeMillis();

//调用 proceed() 方法才会真正的执行实际被代理的方法

Object result = joinPoint.proceed();

long execTimeMillis = System.currentTimeMillis() - startTimeMillis;

logger.info(Method Name : [ + methodName + ] --- AOP method exec time millis : + execTimeMillis);

logger.info(Method Name : [ + methodName + ] --- AOP around end , and result is : + result.toString());

return result;

} catch (Throwable te) {

logger.error(te.getMessage(),te);

throw new RuntimeException(te.getMessage());

}

}

}

首先,需要创建一个类,然后在类名上加上两个注解

@Componentbr/@Aspect

@Component 注解是让这个类被spring当作一个bean管理,@Aspect 注解是标明这个类是一个切面对象

类里面每个方法的注解含义如下:

@Pointcut 用于定义切面的匹配规则,如果想要同事匹配多个的话,可以使用 || 把两个规则连接起来,具体可以参照上面的代码

@Before 目标方法执行前调用

@After 目标方法执行后调用

@AfterReturning 目标方法执行后调用,可以拿到返回结果,执行顺序在 @After 之后

@AfterThrowing 目标方法执行异常时调用

@Around 调用实际的目标方法,可以在目标方法调用前做一些操作,也可以在目标方法调用后做一些操作。使用场景有:事物管理、权限控制,日志打印、性能分析等等

以上就是各个注解的含义和作用,重点的两个注解就是 @Pointcut 和 @Around 注解,@Pointcut用来指定切面规则,决定哪些地方使用这个切面;@Around 会实际的去调用目标方法,这样就可以在目标方法的调用前后做一些处理,例如事物、权限、日志等等。

需要注意的是,这些方法的执行顺序:

执行目标方法前: 先进入 around ,再进入 before

目标方法执行完成后: 先进入 around ,再进入 after ,最后进入 afterreturning

实际的日志信息如下,可以看出各个方法的执行顺序:

另外,使用spring aop 需要在spring的配置文件加上以下这行配置,以开启aop :

aop:aspectj-autoproxy/

同时,maven中需要加入依赖的jar包:

dependency

groupIdorg.aspectj/groupId

artifactIdaspectjrt/artifactId

version1.6.12/version

/dependency

dependency

groupIdorg.aspectj/groupId

artifactIdaspectjweaver/artifactId

version1.6.12/version

/dependency

总结一下,Spring AOP 其实就是使用动态代理来对切面层进行统一的处理,动态代理的方式有:JDK动态代理和 cglib 动态代理,JDK动态代理基于接口实现, cglib 动态代理基于子类实现。spring默认使用的是JDK动态代理,如果没有接口,spring会自动的使用cglib动态代理。

?

原文地址:https://www.cnblogs.com/qfjavabd/p/10175482.html

时间: 2024-10-11 16:03:57

Spring AOP 切面编程记录日志和接口执行时间的相关文章

Spring Aop 切面编程

需求:之前的动态选择数据库,和现在的将某个service层的方法接入cat,都需要用到切面编程. 参考文献: http://www.blogjava.net/supercrsky/articles/174368.html http://my.oschina.net/itblog/blog/211693 一.简介 面向切面编程(AOP)提供另外一种角度来思考程序结构,通过这种方式弥补了面向对象编程(OOP)的不足. 除了类(classes)以外,AOP提供了 切面.切面对关注点进行模块化,例如横切

Spring AOP 切面编程的方法

spring aop的使用分为两种,一种是使用注解来实现,一种是使用配置文件来实现. 先来简单的介绍一下这两种方法的实现,接下来详细的介绍各处的知识点便于查阅.目录如下: 1.基于注解实现spring aop 2.基于配置文件实现spring aop 3.增强的分类 4.切点表达式的构成 1.基于注解来实现的方法 基于注解并不意味着不需要配置文件,只不过是在配置文件中所需要的配置很少.切面编程,显然我们需要一个增强类,一个被增强的普通类. 配置文件中需要添加以下的内容 xmlns:aop="ht

动态代理和AOP切面编程

import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; /* * 动态代理和AOP切面编程 */ //定义一个接口 interface Person { void eat(); void breath(); } // 定义一个实现接口的实现类,即为被代理类 class Student implements Person { @Overr

Spring面向切面编程(二)简单AOP实例

简单实现一个用户登陆的功能,在用户登陆之前进行日志打印,用户登陆之后进行登陆成功日志打印. Maven添加Spring jar spring面向切面编程(一)AOP术语 添加Spring AOP的jar 参考:Maven添加Spring jar 还需添加: 创建User类: package com.user; public class User { private String username; private String password; public String getUsernam

Spring 面向切面编程(AOP)

Spring 系列教程 Spring 框架介绍 Spring 框架模块 Spring开发环境搭建(Eclipse) 创建一个简单的Spring应用 Spring 控制反转容器(Inversion of Control – IOC) 理解依赖注入(DI – Dependency Injection) Bean XML 配置(1)- 通过XML配置加载Bean Bean XML 配置(2)- Bean作用域与生命周期回调方法配置 Bean XML 配置(3)- 依赖注入配置 Bean XML 配置(

SPRING学习(十九)--基于注解的AOP切面编程

上篇中介绍了基于XML配置的AOP切面编程,除了XML配置AOP切面编程外,还可以通过注解方式实现AOP切面编程,本篇通过一个小例子来介绍基于注解的AOP编程. 1.在spring中使用AOP变成,不止要导入spring-aop.jar,还需要导入spring-aspects.jar.aspectjweaver.jar和aopalliance.jar,但是aspectjweaver.jar被spring-aspects.jar依赖,aopalliance.jar被spring-aop.jar依赖

AOP切面编程

AOP的实现底层实际上即为反射,JDK中的反射类java.lang.reflect.Proxy是Java中唯一可以访问调度器的类.类似地,常见的动态代理库cglib也是通过反射机制实现了动态代理的封装.技术成熟度相对较高的AspectJ和Spring AOP会在底层实现一套reflect机制,区别是两者对规范实现如何定义而已. 无论是AspectJ还是Spring AOP,所有这些AOP编程都是基于反射和运行时的动态代理控制实现的.下面通过Proxy和Cglib来实现自己的动态代理切面. 假设我

SpringBoot2.0 基础案例(11):配置AOP切面编程,解决日志记录业务

本文源码 GitHub地址:知了一笑 https://github.com/cicadasmile/spring-boot-base 一.AOP切面编程 1.什么是AOP编程 在软件业,AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术.AOP是OOP(面向对象编程)的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型.利用AOP可以对业务逻辑的各个部

Spring面向切面编程

  1.面向切面编程(AOP)的概念:把项目中需要在多处用到的功能,比如日志.安全和事物等集中到一个类中处理,而不用在每个需要用到该功能的地方显式调用.   2.术语解释:        横切关注点:分布应用于多处的功能        切面:横切关注点可以被模块化为一个类,这个类被称为一个切面        通知(advice):切面要完成的工作.Spring的通知有5种类型:before.after.after-returning.after-throwing和around这五种类型.