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

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

由于使用的是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 需要在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动态代理。

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

  1. Spring AOP 切面编程的方法

    spring aop的使用分为两种,一种是使用注解来实现,一种是使用配置文件来实现. 先来简单的介绍一下这两种方法的实现,接下来详细的介绍各处的知识点便于查阅.目录如下: 1.基于注解实现spring ...

  2. Spring AOP 切面编程实战Demo项目

    为什么会有此项目?在某日,我看博客时,看到了讲面向切面编程的内容,之前也知道spring是面向切面编程的,只是自己没有写过相关的代码,于是决定自己写一个test.但是url拦截器从外部看,和AOP有相 ...

  3. spring aop 切面编程中获取具体方法的方法

    spring 切面编程中获取具体方法的方法 工作中,使用环绕通知,用来捕获异常,然后通过获取方法的返回值,返回不同的数据给到调用方. 由于方法的返回值不同,我们处理异常时,也需要返回不同的格式. 这时 ...

  4. spring aop切面编程实现操作日志步骤

    1.在spring-mvc.xml配置文件中打开切面开关: <aop:aspectj-autoproxy proxy-target-class="true"/> 注意: ...

  5. spring aop 切面编程

    import java.io.IOException; import java.util.ArrayList; import java.util.Arrays; import java.util.Ha ...

  6. Spring MVC通过AOP切面编程 来拦截controller 实现日志的写入

    首选需要参考的是:[参考]http://www.cnblogs.com/guokai870510826/p/5977948.html    http://www.cnblogs.com/guokai8 ...

  7. Spring 面向切面编程(AOP)

    Spring 系列教程 Spring 框架介绍 Spring 框架模块 Spring开发环境搭建(Eclipse) 创建一个简单的Spring应用 Spring 控制反转容器(Inversion of ...

  8. 快速高效掌握企业级项目中的Spring面向切面编程应用,外带讲面试技巧

    Spring面向切面编程(AOP)是企业级应用的基石,可以这样说,如果大家要升级到高级程序员,这部分的知识必不可少. 这里我们将结合一些具体的案例来讲述这部分的知识,并且还将给出AOP部分的一些常见面 ...

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

    本文源码 GitHub地址:知了一笑 https://github.com/cicadasmile/spring-boot-base 一.AOP切面编程 1.什么是AOP编程 在软件业,AOP为Asp ...

随机推荐

  1. iOS 常用小功能 总结

    常用小功能 iOS中的很多小功能都是非常简单的,几行代码就搞定了,比如打电话.打开网址.发邮件.发短信等 打电话 方法一(不被采用): 拨号之前会弹框询问用户是否拨号,拨完后能自动回到原应用 NSUR ...

  2. 转载如何实现portlet之间的传递参数

    Liferay 6开发学习(三十):跨页面Portlet之间的调用与数据传递 2014年10月09日 Liferay 评论 2 条 阅读 4,209 views 次 Portlet之间的通信方法有多种 ...

  3. 植物大战僵尸作弊器源代码(MFC版)

    控制版使用不太方便,此MFC版与控制台版内容一样.具体可以参考前面.此处只附源代码,不加以说明.......... 头文件 // jsMFCDlg.h : 头文件 // #pragma once // ...

  4. 20145316许心远《网络对抗》MSF基础应用

    20145316许心远<网络对抗>MSF基础应用 实验后回答问题 用自己的话解释什么是exploit,payload,encode. exploit:顾名思义就是攻击嘛,因为是个动词,所以 ...

  5. 有趣的js匿名函数写法(function嵌套)

    例子没有什么实际意义,只能做为思路参考 <!DOCTYPE html> <html> <head> <meta charset="UTF-8&quo ...

  6. HTML5 manifest离线缓存技术

    干什么用的? 离线缓存为的是第一次请求后,根据manifest文件进行本地缓存,并且在下一次请求后进行展示(若有缓存的话,无需再次进行请求而是直接调用缓存),最根本的感觉是它使得WEB从online可 ...

  7. DBus学习网站

    http://blog.csdn.net/thonrbirdxb/article/details/11482007 DBus的基本资料可以参考 DBus学习笔记(博客园) http://dotnet. ...

  8. vue-cli使用

    vue-cli 是一个官方发布 vue.js 项目脚手架,使用 vue-cli 可以快速创建 vue 项目,GitHub地址是:https://github.com/vuejs/vue-cli 一.安 ...

  9. oracle中验证身份证是否合法的函数脚本

    --创建函数 入参是身份证   返回1 合法 0不合法 CREATE OR REPLACE FUNCTION fn_checkidcard (p_idcard IN VARCHAR2) RETURN ...

  10. Core Java 1

    p264~p267: 1.程序中可能出现错误:用户输入错误.设备错误.物理限制错误.代码错误 2.如果由于出现错误而使得某些操作没有完成,程序应该:返回一种安全状态,并能够让用户执行一些其他命令: 或 ...