最近客户现在提出系统访问非常慢,需要优化提升访问速度,在排查了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(www.gcyl152.com/)",throwing = "throwable")
  
  public void afterThrowing(JoinPoint joinPoint, Throwable throwable) {
  
  String methodName = joinPoint.getSignature().getName();
  
  logger.info("Method Name :yongshiyule178.com/ [" + 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 : [" www.mcyllpt.com+ methodName + "] ---> AOP around start");
  
  long startTimeMillis = System.currentTimeMillis();
  
  //调用 proceed() 方法才会真正的执行实际被代理的方法
  
  Object result = joinPoint.proceed();
  
  long execTimeMillis = System.currentTimeMillis() - startTimeMillis;
  
  logger.info("Method Name : [" + methodName + "] -www.dasheng178.com/--> AOP method exec time millis : " + execTimeMillis);
  
  logger.info("Method Name : www.yigouyule2.cn [" + methodName + "] ---> AOP around end , and result is : " + result.toString());
  
  return result;
  
  } catch (Throwable te) {
  
  logger.error(te.getMessage(),te);
  
  throw new RuntimeException(te.getMessage());
  
  }
  
  }
  
  }
  
  首先,需要创建一个类,然后在类名上加上两个注解
  
  @Component
  
  @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>
  
  <groupId>org.aspectj</groupId>
  
  <artifactId>aspectjrt</artifactId>
  
  <version>1.6.12</version>
  
  </dependency>
  
  <dependency>
  
  <groupId>org.aspectj</groupId>
  
  <artifactId>aspectjweaver</artifactId>
  
  <version>1.6.12</version>
  
  </dependency>
  
  总结一下,Spring AOP 其实就是使用动态代理来对切面层进行统一的处理,动态代理的方式有:JDK动态代理和 cglib 动态代理,JDK动态代理基于接口实现, cglib 动态代理基于子类实现。spring默认使用的是JDK动态代理,如果没有接口,spring会自动的使用cglib动态代理。

排查nginx、tomcat内存和服务器负载之后的更多相关文章

  1. Nginx + Tomcat Windows下的负载均衡配置

     Nginx + Tomcat Windows下的负载均衡配置 一.为什么需要对Tomcat服务器做负载均衡?    Tomcat服务器作为一个Web服务器,其并发数在300-500之间,如果超过50 ...

  2. 图文解说:Nginx+tomcat配置集群负载均衡

    图文解说:Nginx+tomcat配置集群负载均衡 博客分类: appserver nginxTomcatUbuntuLinux网络应用  作者:niumd Blog:http://ari.iteye ...

  3. 搭建 Keepalived + Nginx + Tomcat 的高可用负载均衡架构

    1 概述 初期的互联网企业由于业务量较小,所以一般单机部署,实现单点访问即可满足业务的需求,这也是最简单的部署方式,但是随着业务的不断扩大,系统的访问量逐渐的上升,单机部署的模式已无法承载现有的业务量 ...

  4. Keepalived + Nginx + Tomcat 的高可用负载均衡架构搭建

    Keepalived + Nginx + Tomcat 的高可用负载均衡架构搭建 Nginx 是一个高性能的 HTTP反向代理服务器 Keepalived 是一个基于VRRP协议来实现的LVS服务高可 ...

  5. 搭建Keepalived + Nginx + Tomcat的高可用负载均衡架构

    1 概述 初期的互联网企业由于业务量较小,所以一般单机部署,实现单点访问即可满足业务的需求,这也是最简单的部署方式,但是随着业务的不断扩大,系统的访问量逐渐的上升,单机部署的模式已无法承载现有的业务量 ...

  6. Nginx+Tomcat多实例及负载均衡配置

    Nginx+Tomcat多实例及负载均衡配置 采用nginx的反向代理负载均衡功能,配合后端的tomcat多实例来实现tomcat WEB服务的负载均衡 01 安装nginx服务 安装所需的pcre库 ...

  7. nginx+tomcat+memcached搭建服务器集群及负载均衡

    在实际项目中,由于用户的访问量很大的原因,往往需要同时开启多个服务器才能满足实际需求.但是同时开启多个服务又该怎么管理他们呢?怎样实现session共享呢?下面就来讲一讲如何使用tomcat+ngin ...

  8. Keepalived+Nginx+Tomcat配置高可用负载均衡系统示例

    前言 此示例为keepalived+nginx+tomcat的基础配置示例,某些特定配置此例中不会出现,在示例中会用到三个虚拟机:两个纯命令行用于模拟服务端配置,一个带桌面环境的用于模拟客户端访问,这 ...

  9. Nginx+tomcat配置集群负载均衡

    开发的应用采用F5负载均衡交换机,F5将请求转发给5台hp unix服务器,每台服务器有多个webserver实例,对外提供web服务和socket等接口服务.之初,曾有个小小的疑问为何不采用开源的a ...

随机推荐

  1. 【ntp时间校准配置】

    Ntp(网络时间协议)是一种可以通过TCP/IP网络传播,其架构模式可分为C/S(客户端/服务器),PTP(对等),broatcast(广播), mutilbrocast(组播),无论在任何系统或设备 ...

  2. 关于Ext.js和Ext.Net的杂谈

    最近几年比较火的前端js框架extjs 算是其中的佼佼者.统一的UI设计,强悍的组件及丰富的插件,对浏览器良好的兼容性等优点使得许多公司使用Extjs,同时也使得无数程序猿开始研究这个玩意也包括我在内 ...

  3. 电子商城实录------定义init初始化的方法

    路由方法的设置 //路由方法 private static function dispatch(){ //获取控制器名称(类比:英文单词的后缀) $controller_name=CONTROLLER ...

  4. java.lang.UnsupportedOperationException: seccomp unavailable: CONFIG_SECCOMP not compiled into kernel, CONFIG_SECCOMP and CONFIG_SECCOMP_FILTER are needed

    错误描述: ElasticSearch集群启动错误,错误的原因是:因为Centos6不支持SecComp,而ES默认bootstrap.system_call_filter为true进行检测,所以导致 ...

  5. phpcms2008网站漏洞如何修复 远程代码写入缓存漏洞利用

    SINE安全公司在对phpcms2008网站代码进行安全检测与审计的时候发现该phpcms存在远程代码写入缓存文件的一个SQL注入漏洞,该phpcms漏洞危害较大,可以导致网站被黑,以及服务器遭受黑客 ...

  6. Altera Stratix IV 命名规则

    由于要开发基于DE4平台的应用,应该要了解一下该平台的芯片情况Stratix IV 具体型号为:Stratix IV EP4SGX230KF40C2 命名规范如下 官网资料为:https://www. ...

  7. linux c fprintf()

    #include<stdio.h> #include<unistd.h> #include<time.h> int main(int argc,char *argv ...

  8. stm32--FatFs移植(SPIFlash)

    前言 硬件: 单片机:stm32f072CB,sram大小16k.(其他单片机只要sram>8k即可通用) SPIFlash:W25Q128FV,16Mbyte,单次擦除最小4k. 程序使用Ke ...

  9. 为什么在默认情况下无法修改被block捕获的变量? __block都做了什么?

    默认情况下,block里面的变量,拷贝进去的是变量的值,而不是指向变量的内存的指针.使用__block修饰后的变量,拷贝到block里面的就是指向变量的指针,所以我们就可以修改变量的值.

  10. Android开发免费类库和工具集合

    用于Android开发的免费类库和工具集合,按目录分类. Action Bars ActionBarSherlock Extended ActionBar FadingActionBar GlassA ...