Spring计时器stopwatch使用详解

网友投稿 1894 2022-12-19

Spring计时器stopwatch使用详解

Spring计时器stopwatch使用详解

StopWatch是位于org.springframework.util包下的一个工具类,通过它可方便的对程序部分代码进行计时(ms级别),适用于同步单线程代码块。

正常情况下,我们如果需要看某段代码的执行耗时,会通过如下的方式进行查看:

public static void main(String[] args) throws InterruptedException {

StopWatchTest.test0();

// StopWatchTest.test1();

}

public static void test0() throws InterruptedException {

long start = System.currentTimeMillis();

http:// // do something

Thread.sleep(100);

long end = System.currentTimeMillis();

long start2 = System.currentTimeMillis();

// do something

Thread.sleep(200);

long end2 = System.currentTimeMillis();

System.out.println("某某1执行耗时:" + (end - start));

System.out.println("某某2执行耗时:" + (end2 - start2));

}

运行结果:

某某1执行耗时:105

某某2执行耗时:203

该种方法通过获取执行完成时间与执行开始时间的差值得到程序的执行时间,简单直接有效,但想必写多了也是比较烦人的,尤其是碰到不可描述的代码时,会更加的让人忍不住多写几个bug聊表敬意,而且该结果也不够直观,此时会想是否有一个工具类,提供了这些方法,或者自己写个工具类,刚好可以满足这种场景,并且把结果更加直观的展现出来。

首先我们的需求如下:

记录开始时间点

记录结束时间点

输出执行时间及各个时间段的占比

根据该需求,我们可直接使用org.springframework.util包下的一个工具类StopWatch,通过该工具类,我们对上述代码做如下改造:

public static void main(String[] args) throws InterruptedException {

// StopWatchTest.test0();

StopWatchTest.test1();

}

public static void test1() throws InterruptedException {

StopWatch sw = new StopWatch("test");

sw.start("task1");

// do something

Thread.sleep(100);

sw.stop();

sw.start("task2");

// do something

Thread.sleep(200);

sw.stop();

System.out.println("sw.prettyPrint()~~~~~~~~~~~~~~~~~");

System.out.println(sw.prettyPrint());

}

运行结果:

sw.prettyPrint()~~~~~~~~~~~~~~~~~

StopWatch 'test': running time (millis) = 308

-----------------------------------------

ms % Task name

-----------------------------------------

00104 034% task1

00204 066% task2

start开始记录,stop停止记录,然后通过StopWatch的prettyPrint方法,可直观的输出代码执行耗时,以及执行时间百分比,瞬间感觉比之前的方式高大上了一个档次。

除此之外,还有以下两个方法shortSummary,getTotalTimeMillis,查看程序执行时间。

运行代码及结果:

System.out.println("sw.shortSummary()~~~~~~~~~~~~~~~~~");

System.out.println(sw.shortSummary());

System.out.println("sw.getTotalTimeMillis()~~~~~~~~~~~~~~~~~");

System.out.println(sw.getTotalTimeMillis());

运行结果

sw.shortSummary()~~~~~~~~~~~~~~~~~

StopWatch 'test': running time (millis) = 308

sw.getTotalTimeMillis()~~~~~~~~~~~~~~~~~

308

其实以上内容在该工具类中实现也极其简单,通过start与stop方法分别记录开始时间与结束时间,其中在记录结束时间时,会维护一个链表类型的tasklist属性,从而使该类可记录多个任务,最后的输出也仅仅是对之前记录的信息做了一个统一的归纳输出,从而使结果更加直观的展示出来。

StopWatch优缺点:

优点:

spring自带工具类,可直接使用

代码实现简单,使用更简单

统一归纳,展示每项任务耗时与占用总时间的百分比,展示结果直观性能消耗相对较小,并且最大程度的保证了start与stop之间的时间记录的准确性

可在start时直接指定任务名字,从而更加直观的显示记录结果

缺点:

一个StopWatch实例一次只能开启一个task,不能同时start多个task,并且在该task未stop之前不能start一个新的task,必须在该task stop之后才能开启新的task,若要一次开启多个,需要new不同的StopWatch实例

代码侵入式使用,需要改动多处代码

spring中StopWatch源码实现如下:

import java.text.NumberFormat;

import java.util.LinkedList;

import java.util.List;

public class StopWatch {

private final String id;

private boolean keepTaskList = true;

private final List taskList = new LinkedList();

private long startTimeMillis;

private boolean running;

private String currentTaskName;

private StopWatch.TaskInfo lastTaskInfo;

private int taskCount;

private long totalTimeMillis;

public StopWatch() {

this.id = "";

}

public StopWatch(String id) {

this.id = id;

}

public void setKeepTaskList(boolean keepTaskList) {

this.keepTaskList = keepTaskList;

}

public void start() throws IllegalStateException {

this.start("");

}

public void start(String taskName) throws IllegalStateException {

if (this.running) {

throw new IllegalStateException("Can't start StopWatch: it's already running");

} else {

this.startTimeMillis = System.currentTimeMillis();

this.running = true;

this.currentTaskName = taskName;

}

}

public void stop() throws IllegalStateException {

if (!this.running) {

throw new IllegalStateException("Can't stop StopWatch: it's not running");

} else {

long lastTime = System.currentTimeMillis() - this.startTimeMillis;

this.totalTimeMillis += lastTime;

this.lastTaskInfo = new StopWatch.TaskInfo(this.currentTaskName, lastTime);

if (this.keepTaskList) {

this.taskList.add(this.lastTaskInfo);

}

++this.taskCount;

this.running = false;

this.currentTaskName = null;

}

}

public boolean isRunning() {

return this.running;

}

public long getLastTaskTimeMillis() throws IllegalStateException {

if (this.lastTaskInfo == null) {

throw new IllegalStateException("No tasks run: can't get last task interval");

} else {

return this.lastTaskInfo.getTimeMillis();

}

}

public String getLastTaskName() throws IllegalStateException {

if (this.lastTaskInfo == null) {

throw new IllegalStateException("No tasks run: can't get last task name");

} else {

return this.lastTaskInfo.getTaskName();

}

}

public StopWatch.TaskInfo getLastTaskInfo() throws IllegalStateException {

if (this.lastTaskInfo == null) {

throw new IllegalStateException("No tasks run: can't get last task info");

} else {

return this.lastTaskInfo;

}

}

public long getTotalTimeMillis() {

return this.totalTimeMillis;

}

public double getTotalTimeSeconds() {

return (double) this.totalTimeMillis / 1000.0D;

}

public int getTaskCount() {

return this.taskCount;

}

public StopWatch.TaskInfo[] getTaskInfo() {

if (!this.keepTaskList) {

throw new UnsupportedOperationException("Task info is not being kept!");

} else {

http:// return (StopWatch.TaskInfo[]) this.taskList.toArray(new StopWatch.TaskInfo[this.taskList.size()]);

}

}

public String shortSummary() {

return "StopWatch '" + this.id + "': running time (millis) = " + this.getTotalTimeMillis();

}

public String prettyPrint() {

StringBuilder sb = new StringBuilder(this.shortSummary());

sb.append('\n');

if (!this.keepTaskList) {

sb.append("No task info kept");

} else {

sb.append("-----------------------------------------\n");

sb.append("ms % Task name\n");

sb.append("-----------------------------------------\n");

NumberFormat nf = NumberFormat.getNumberInstance();

nf.setMinimumIntegerDigits(5);

nf.setGroupingUsed(false);

NumberFormat pf = NumberFormat.getPercentInstance();

pf.setMinimumIntegerDigits(3);

pf.setGroupingUsed(false);

StopWatch.TaskInfo[] var7;

int var6 = (var7 = this.getTaskInfo()).length;

for (int var5 = 0; var5 < var6; ++var5) {

StopWatch.TaskInfo task = var7[var5];

sb.append(nf.format(task.getTimeMillis())).append(" ");

sb.append(pf.format(task.getTimeSeconds() / this.getTotalTimeSeconds())).append(" ");

sb.append(task.getTaskName()).append("\n");

}

}

return sb.toString();

}

@Override

public String toString() {

StringBuilder sb = new StringBuilder(this.shortSummary());

if (this.keepTaskList) {

StopWatch.TaskInfo[] var5;

int var4 = (var5 = this.getTaskInfo()).length;

for (int var3 = 0; var3 < var4; ++var3) {

StopWatch.TaskInfo task = var5[var3];

sb.append("; [").append(task.getTaskName()).append("] took ").append(task.getTimeMillis());

long percent = Math.round(100.0D * task.getTimeSeconds() / this.getTotalTimeSeconds());

sb.append(" = ").append(percent).append("%");

}

} else {

sb.append("; no task info kept");

}

return sb.toString();

}

public static final class TaskInfo {

private final String taskName;

private final long timeMillis;

Thttp://askInfo(String taskName, long timeMillis) {

this.taskName = taskName;

this.timeMillis = timeMillis;

}

public String getTaskName() {

return this.taskName;

}

public long getTimeMillis() {

return this.timeMillis;

}

public double getTimeSeconds() {

return (double) this.timeMillis / 1000.0D;

}

}

}

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:springmvc字符编码过滤器CharacterEncodingFilter的使用
下一篇:SpringBoot开发实战系列之动态定时任务
相关文章

 发表评论

暂时没有评论,来抢沙发吧~