jsp中java代码无效_来杯咖啡,教你如何优雅的在java中统计代码块耗时

推荐阅读:

Sping源码+Redis+Nginx+MySQL等七篇实战技术文档,阿里大佬推荐

阿里内部:2020年全技术栈文档+PPT分享,(万粉总结,回馈粉丝)

1e0264bc6a52929e72f946c03636eb9f.png

在我们的实际开发中,多多少少会遇到统计一段代码片段的耗时的情况,我们一般的写法如下

long start = System.currentTimeMillis();try {    // .... 具体的代码段} finally {    System.out.println("cost: " + (System.currentTimeMillis() - start));}

上面的写法没有什么毛病,但是看起来就不太美观了,那么有没有什么更优雅的写法呢?

1. 代理方式

了解 Spring AOP 的同学可能立马会想到一个解决方法,如果想要统计某个方法耗时,使用切面可以无侵入的实现,如

// 定义切点,拦截所有满足条件的方法@Pointcut("execution(public * com.git.hui.boot.aop.demo.*.*(*))")public void point() {}@Around("point()")public Object doAround(ProceedingJoinPoint joinPoint) throws Throwable {    long start = System.currentTimeMillis();    try{        return joinPoint.proceed();    } finally {        System.out.println("cost: " + (System.currentTimeMillis() - start));    }}

Spring AOP 的底层支持原理为代理模式,为目标对象提供增强功能;在 Spring 的生态体系下,使用 aop 的方式来统计方法耗时,可以说少侵入且实现简单,但是有以下几个问题

  • 统计粒度为方法级别
  • 类内部方法调用无法生效(详情可以参考博文:【SpringBoot 基础系列教程】AOP 之高级使用技能)

2. AutoCloseable

在 JDK1.7 引入了一个新的接口AutoCloseable, 通常它的实现类配合try{}使用,可在 IO 流的使用上,经常可以看到下面这种写法

// 读取文件内容并输出try (Reader stream = new BufferedReader(new InputStreamReader(new FileInputStream("/tmp")))) {    List list = ((BufferedReader) stream).lines().collect(Collectors.toList());    System.out.println(list);} catch (IOException e) {    e.printStackTrace();}

注意上面的写法中,最值得关注一点是,不需要再主动的写stream.close了,主要原因就是在try(){}执行完毕之后,会调用方法AutoCloseable#close方法;

基于此,我们就会有一个大单的想法,下一个Cost类实现AutoCloseable接口,创建时记录一个时间,close 方法中记录一个时间,并输出时间差值;将需要统计耗时的逻辑放入try(){}代码块

下面是一个具体的实现:

public static class Cost implements AutoCloseable {    private long start;    public Cost() {        this.start = System.currentTimeMillis();    }    @Override    public void close() {        System.out.println("cost: " + (System.currentTimeMillis() - start));    }}public static void testPrint() {    for (int i = 0; i < 5; i++) {        System.out.println("now " + i);        try {            Thread.sleep(10);        } catch (InterruptedException e) {            e.printStackTrace();        }    }}public static void main(String[] args) {    try (Cost c = new Cost()) {        testPrint();    }    System.out.println("------over-------");}

执行后输出如下:

now 0now 1now 2now 3now 4cost: 55

如果代码块抛异常,也会正常输出耗时么?

public static void testPrint() {    for (int i = 0; i < 5; i++) {        System.out.println("now " + i);        try {            Thread.sleep(10);        } catch (InterruptedException e) {            e.printStackTrace();        }        if (i == 3) {            throw new RuntimeException("some exception!");        }    }}

再次输出如下,并没有问题

now 0now 1now 2now 3cost: 46Exception in thread "main" java.lang.RuntimeException: some exception!at com.git.hui.boot.order.Application.testPrint(Application.java:43)at com.git.hui.boot.order.Application.main(Application.java:50)

3. 小结

除了上面介绍的两种方式,还有一种在业务开发中不太常见,但是在中间件、偏基础服务的功能组件中可以看到,利用 Java Agent 探针技术来实现,比如阿里的 arthas 就是在 JavaAgent 的基础上做了各种上天的功能,后续介绍 java 探针技术时会专门介绍

下面小结一下三种统计耗时的方式

基本写法

long start = System.currentTimeMillis();try {    // .... 具体的代码段} finally {    System.out.println("cost: " + (System.currentTimeMillis() - start));}

优点是简单,适用范围广泛;缺点是侵入性强,大量的重复代码

Spring AOP

在 Spring 生态下,可以借助 AOP 来拦截目标方法,统计耗时

@Around("...")public Object doAround(ProceedingJoinPoint joinPoint) throws Throwable {    long start = System.currentTimeMillis();    try{        return joinPoint.proceed();    } finally {        System.out.println("cost: " + (System.currentTimeMillis() - start));    }}

优点:无侵入,适合统一管理(比如测试环境输出统计耗时,生产环境不输出);缺点是适用范围小,且粒度为方法级别,并受限于 AOP 的使用范围

AutoCloseable

这种方式可以看做是第一种写法的进阶版

// 定义类public static class Cost implements AutoCloseable {    private long start;    public Cost() {        this.start = System.currentTimeMillis();    }    @Override    public void close() {        System.out.println("cost: " + (System.currentTimeMillis() - start));    }}// 使用姿势try (Cost c = new Cost()) {    ...}

优点是:简单,适用范围广泛,且适合统一管理;缺点是依然有代码侵入

说明

上面第二种方法看着属于最优雅的方式,但是限制性强;如果有更灵活的需求,建议考虑第三种写法,在代码的简洁性和统一管理上都要优雅很多,相比较第一种可以减少大量冗余代码


作者:一灰灰
链接:https://juejin.im/post/5e5e4cc6518825493d6a96c4

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/561005.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

linux 随机抽取文件,shell 随机从文件中抽取若干行的实现方法

shuf -n5 main.txtsort -R main.txt | head -5awk -vN5 -vC"wc -l file" BEGIN{srand();while(nwhile read line;do echo "$line $RANDOM";done < main.txt | sort -k2,2n| awk NR<5{print $1}shuf 命令的选项&#xff1a;-e, --echo &#xff1a;将…

c++—引用。。。

目录 目录前言引用的概念交换a和b的值&#xff1a;引用作为函数的返回值常引用例子 前言 纯自学&#xff0c;看的mooc上郭老师的课&#xff0c;整理整理。。。不然全忘了。。 引用的概念 类型名&引用名某变量名&#xff1b; int&rn&#xff1b; r引用了n&#x…

服务器闰秒 linux,闰秒导致部分 Linux 服务器高 CPU 使用率

国际地球自转和参考坐标系统服务(IERS)在2012年6月30日午夜(北京时间7月1号7点59分59秒)增加一闰秒(即出现7&#xff1a;59&#xff1a;60)。由于Linux kernel 2.6.29之前版本存在bug&#xff0c;在进行闰秒调整时可能会引起系统时钟服务ntpd进程死锁。Debian Lenny、RHEL/Cent…

const

目录 目录定义常量定义常量指针定义常引用 定义常量 const int max22&#xff1b; const double p3.1415; const char* sch"zzuli"; 定义常量指针 1.不可通过常量指针修改其指向的内容 &#xff5b;常量指针指向的内容不能随意修改&#xff5d; int n,m; cons…

http header 设置编码_【译】http.client

本模块实现了HTTP和HTTPS协议的客户端功能。通常本模块不会被直接使用&#xff0c;而是被urllib.request调用&#xff0c;来处理HTTP和HTTPS相关的URL。备注&#xff1a;HTTPS只有在支持SSL(带有ssl模块)的Python编译器里面才是可用的。(一)模块提供的类class http.client.HTTP…

diskgenius linux 分区,DiskGenius怎么分区,DiskGenius分区教程

DiskGenius分区教程方法一&#xff1a;一、首先&#xff0c;打开DiskGenius软件并单击以选择要操作的硬盘驱动器。二、然后右键单击“快速分区”按钮并单击“确定”。方法2&#xff1a;三、选择要操作的硬盘&#xff0c;单击软件上方的硬盘按钮&#xff0c;单击下面列中的快速分…

new

动态内存分配 用new运算符实现动态内存分配 动态内存分配分配一个变量分配一个数组new运算符返回值类型释放动态分配出来的内存空间动态分配指针数组 分配一个变量 pnew t; t是任意类型名&#xff0c;p为类型为t*的指针。 动态分配出一片大小为sizeof(t)字节的内存空间&a…

linux系统ip占用,IP地址被占用的问题,折腾我好几天了 (已解决)

原因找到了&#xff0c;另外一台机器的网卡上面不知道怎么绑定了两个IP地址eth0 Link encap:Ethernet HWaddr 00:20:ED:1A:62:7Einet addr:192.168.0.104 Bcast:192.168.0.255 Mask:255.255.255.0inet6 addr: fe80::220:edff:fe1a:627e/64 Scope:LinkUP BROADCAST RUNN…

sql server management studio性能分析_如何分析一条SQL的性能

来自公众号&#xff1a;谭小谭这篇文章将给大家介绍如何使用 explain 来分析一条 sql 。网上其实已经有非常多的文章都很详细的介绍了 explain 的使用&#xff0c;这篇文章将实例和原理结合起来&#xff0c;尽量让你有更好的理解&#xff0c;相信我&#xff0c;认真看完你应该会…

malloc动态分配数组以及指针数组大小

对于二维数组&#xff1a;一般来说&#xff0c;当给两个数分别表示行列&#xff0c;但两个数的范围过大&#xff0c;就需要动态分配。 int i,j,m,n;scanf("%d%d",&m,&n);int **a;a(int**)malloc(sizeof(int*)*m);for(i0;i<m;i)a[i](int *)malloc(sizeof(i…

编译运行linux0.12,linux0.12 编译过程

感谢这篇文章的作者&#xff1a; http://www.cnblogs.com/strugglesometimes/p/4231359.html编译是个很蛋疼的事情&#xff0c;本想把linux0.12在bochs上跑起来然后就可以各模块的学习&#xff0c;没想各种问题。问题1&#xff1a;1 gas -c -o boot/head.o boot/head.s2 mak…

不用数组,解决众数问题(前提 :众数出现的次数必须大于n/2)

内存限制4mb 用数组不通过&#xff0c;怎么办&#xff1f; 众数出现的次数必须大于n/2 第一行输入一个整数n &#xff08;1<n<1E6) 接下来一行n个整数 mi &#xff08;1<MI<1E9) 表示第i种糖果的个数&#xff0c;整数之间用空格隔开 输出mi中出现次数最多的那…

高考python必考题目_假如高考考python编程,这些题目你会几个呢?

Python(发音&#xff1a;英[?pa?θ?n]&#xff0c;美[?pa?θɑ:n])&#xff0c;是一种面向对象、直译式电脑编程语言&#xff0c;也是一种功能强大的通用型语言&#xff0c;已经具有近二十年的发展历史&#xff0c;成熟且稳定。它包含了一组完善而且容易理解的标准库&…

编译linux tq2440,QT4.8.2在TQ2440开发板上的移植(一)--编译和安装

主机版本&#xff1a;Ubuntu 11.04交叉编译器版本&#xff1a;4.3.3移植的主要工作就是编译在ARM板上运行的qt库&#xff0c;并且把这些库做到根目录中。需要的文件tslib-1.4.tar.gz qt-everywhere-opensource-src-4.8.2.tar.gz具体步骤如下&#xff1a;1、首先编译安装触摸屏驱…

图书管理系统_目前图书管理系统存在的问题

作者&#xff1a;新风学术网(一) 不能准确直观地指明图书所在的空间位置目前所使用的管理系统在索书的过程中是读者先在图书馆查询系统上查询到所要借的图书并记录下这本书的索书号和馆藏地, 再根据索书号到书的馆藏地所在位置查找书。有些读者对索书号是怎么排架的并不了解, 也…

python怎么计算图像梯度_opencv python图像梯度实例详解

这篇文章主要介绍了opencv python图像梯度实例详解,文中通过示例代码介绍的非常详细&#xff0c;对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下一阶导数与Soble算子二阶导数与拉普拉斯算子图像边缘&#xff1a;Soble算子&#xff1a;二阶导数&#xff1a;拉…

vector简单应用

输出vector中元素&#xff0c;以及插入删除元素 #include<iostream> #include<vector> using namespace std; template<class T> void printvector(T s,T e)//输出vector元素 {for(; s!e; s)cout<<*s<<" ";cout<<endl; } int …

Linux ct获取本机ip,linux ip命令

ip 是个命令&#xff0c; ip 命令的功能很多&#xff01;基本上它整合了 ifconfig 与 route 这两个命令&#xff0c;不过ip 的功能更强大&#xff01;如果您有兴趣的话&#xff0c;请自行 vi /sbin/ifup 就知道整个 ifup 就是利用 ip这个命令来实现的。下面介绍一下使用方法[ro…

linux errno的作用域,关于比特科技c语言的学习博客(1)

写代码1创建工程2创建路径3创建源文件4写代码写c代码时.c文件是源文件.h是头文件写helloworld时return 0记得中间敲空格 main是主函数从main开始执行也是程序的入口有且仅有一个int是整型的意思main前的int表示main函数调用返回一个整型值void main已经过时#include 包含一个叫…