Java

前言

代码耗时统计在日常开发中算是一个十分常见的需求,特别是在需要找出代码性能瓶颈时。
可能也是受限于 Java 的语言特性,总觉得代码写起来不够优雅,大量的耗时统计代码,干扰了业务逻辑。特别是开发功能的时候,有个感受就是刚刚开发完代码很清爽优雅,结果加了一大堆辅助代码后,整个代码就变得臃肿了,自己看着都挺难受。因此总想着能不能把这块写的更优雅一点,今天本文就尝试探讨下“代码耗时统计”这一块。
在开始正文前,先说下前提,“代码耗时统计”的并不是某个方法的耗时,而是任意代码段之间的耗时。这个代码段,可能是一个方法中的几行代码,也有可能是从这个方法的某一行到另一个被调用方法的某一行,因此通过 AOP 方式是不能实现这个需求的。

常规方法

时间差统计

这种方式是最简单的方法,记录下开始时间,再记录下结束时间,计算时间差即可。

  1. public class TimeDiffTest {
  2. public static void main(String[] args) throws InterruptedException {
  3. final long startMs = TimeUtils.nowMs();
  4. TimeUnit.SECONDS.sleep(5); // 模拟业务代码
  5. System.out.println("timeCost: " + TimeUtils.diffMs(startMs));
  6. }
  7. }
  8. /* output:
  9. timeCost: 5005
  10. */
  11. public class TimeUtils {
  12. /**
  13. * @return 当前毫秒数
  14. */
  15. public static long nowMs() {
  16. return System.currentTimeMillis();
  17. }
  18. /**
  19. * 当前毫秒与起始毫秒差
  20. * @param startMillis 开始纳秒数
  21. * @return 时间差
  22. */
  23. public static long diffMs(long startMillis) {
  24. return diffMs(startMillis, nowMs());
  25. }
  26. }

这种方式的优点是实现简单,利于理解;缺点就是对代码的侵入性较大,看着很傻瓜,不优雅。

StopWatch

第二种方式是参考 StopWatch ,StopWatch 通常被用作统计代码耗时,各个框架和 Common 包都有自己的实现。

  1. public class TraceWatchTest {
  2. public static void main(String[] args) throws InterruptedException {
  3. TraceWatch traceWatch = new TraceWatch();
  4. traceWatch.start("function1");
  5. TimeUnit.SECONDS.sleep(1); // 模拟业务代码
  6. traceWatch.stop();
  7. traceWatch.start("function2");
  8. TimeUnit.SECONDS.sleep(1); // 模拟业务代码
  9. traceWatch.stop();
  10. traceWatch.record("function1", 1); // 直接记录耗时
  11. System.out.println(JSON.toJSONString(traceWatch.getTaskMap()));
  12. }
  13. }
  14. /* output:
  15. {"function2":[{"data":1000,"taskName":"function2"}],"function1":[{"data":1000,"taskName":"function1"},{"data":1,"taskName":"function1"}]}
  16. */
  17. public class TraceWatch {
  18. /** Start time of the current task. */
  19. private long startMs;
  20. /** Name of the current task. */
  21. @Nullable
  22. private String currentTaskName;
  23. @Getter
  24. private final Map<String, List<TaskInfo>> taskMap = new HashMap<>();
  25. /**
  26. * 开始时间差类型指标记录,如果需要终止,请调用 {@link #stop()}
  27. *
  28. * @param taskName 指标名
  29. */
  30. public void start(String taskName) throws IllegalStateException {
  31. if (this.currentTaskName != null) {
  32. throw new IllegalStateException("Can't start TraceWatch: it's already running");
  33. }
  34. this.currentTaskName = taskName;
  35. this.startMs = TimeUtils.nowMs();
  36. }
  37. /**
  38. * 终止时间差类型指标记录,调用前请确保已经调用
  39. */
  40. public void stop() throws IllegalStateException {
  41. if (this.currentTaskName == null) {
  42. throw new IllegalStateException("Can't stop TraceWatch: it's not running");
  43. }
  44. long lastTime = TimeUtils.nowMs() - this.startMs;
  45. TaskInfo info = new TaskInfo(this.currentTaskName, lastTime);
  46. this.taskMap.computeIfAbsent(this.currentTaskName, e -> new LinkedList<>()).add(info);
  47. this.currentTaskName = null;
  48. }
  49. /**
  50. * 直接记录指标数据,不局限于时间差类型
  51. * @param taskName 指标名
  52. * @param data 指标数据
  53. */
  54. public void record(String taskName, Object data) {
  55. TaskInfo info = new TaskInfo(taskName, data);
  56. this.taskMap.computeIfAbsent(taskName, e -> new LinkedList<>()).add(info);
  57. }
  58. @Getter
  59. @AllArgsConstructor
  60. public static final class TaskInfo {
  61. private final String taskName;
  62. private final Object data;
  63. }
  64. }

这是仿照 org.springframework.util.StopWatch 的实现,写了 TraceWatch 类,这个方法提供了两种耗时统计的方法:
通过调用 Start(name)Stop() 方法,进行耗时统计。通过调用 Record(name, timeCost),方法,直接记录耗时信息。这种方式本质上和“时间差统计”是一致的,只是抽取了一层,稍微优雅了一点。
注:可以根据自己的业务需要,自行修改 TraceWatch 内部的数据结构,这里简单起见,内部的数据结构只是随便举了个例子。

高级方法

第二节提到的两种方法,用大白话来说都是“直来直去”的感觉,还可以尝试把代码写的更简便一点。

Function

在 jdk 1.8 中,引入了 java.util.function 包,通过该类提供的接口,能够实现在指定代码段的上下文执行额外代码的功能。

  1. public class TraceHolderTest {
  2. public static void main(String[] args) {
  3. TraceWatch traceWatch = new TraceWatch();
  4. TraceHolder.run(traceWatch, "function1", i -> {
  5. try {
  6. TimeUnit.SECONDS.sleep(1); // 模拟业务代码
  7. } catch (InterruptedException e) {
  8. e.printStackTrace();
  9. }
  10. });
  11. String result = TraceHolder.run(traceWatch, "function2", () -> {
  12. try {
  13. TimeUnit.SECONDS.sleep(1); // 模拟业务代码
  14. return "YES";
  15. } catch (InterruptedException e) {
  16. e.printStackTrace();
  17. return "NO";
  18. }
  19. });
  20. TraceHolder.run(traceWatch, "function1", i -> {
  21. try {
  22. TimeUnit.SECONDS.sleep(1); // 模拟业务代码
  23. } catch (InterruptedException e) {
  24. e.printStackTrace();
  25. }
  26. });
  27. System.out.println(JSON.toJSONString(traceWatch.getTaskMap()));
  28. }
  29. }
  30. /* output:
  31. {"function2":[{"data":1004,"taskName":"function2"}],"function1":[{"data":1001,"taskName":"function1"},{"data":1002,"taskName":"function1"}]}
  32. */
  33. public class TraceHolder {
  34. /**
  35. * 有返回值调用
  36. */
  37. public static <T> T run(TraceWatch traceWatch, String taskName, Supplier<T> supplier) {
  38. try {
  39. traceWatch.start(taskName);
  40. return supplier.get();
  41. } finally {
  42. traceWatch.stop();
  43. }
  44. }
  45. /**
  46. * 无返回值调用
  47. */
  48. public static void run(TraceWatch traceWatch, String taskName, IntConsumer function) {
  49. try {
  50. traceWatch.start(taskName);
  51. function.accept(0);
  52. } finally {
  53. traceWatch.stop();
  54. }
  55. }
  56. }

这里利用了 SupplierIntConsumer 接口,对外提供了有返回值和无返回值得调用,在 TraceHolder 类中,在核心代码块的前后,分别调用了前文的 TraceWatch 的方法,实现了耗时统计的功能。

AutoCloseable

除了利用 Function 的特性,还可以使用 jdk 1.7 的 AutoCloseable 特性。说 AutoCloseable 可能有同学没听过,但是给大家展示下以下代码,就会立刻明白是什么东西了。

  1. // 未使用 AutoCloseable
  2. public static String readFirstLingFromFile(String path) throws IOException {
  3. BufferedReader br = null;
  4. try {
  5. br = new BufferedReader(new FileReader(path));
  6. return br.readLine();
  7. } catch (IOException e) {
  8. e.printStackTrace();
  9. } finally {
  10. if (br != null) {
  11. br.close();
  12. }
  13. }
  14. return null;
  15. }
  16. // 使用 AutoCloseable
  17. public static String readFirstLineFromFile(String path) throws IOException {
  18. try (BufferedReader br = new BufferedReader(new FileReader(path))) {
  19. return br.readLine();
  20. }
  21. }

try 后方可以加载一个实现了 AutoCloseable 接口的对象,该对象作用于整个 try 语句块中,并且在执行完毕后回调 AutoCloseable#close() 方法。
对 TraceWatch 类进行改造:
实现 AutoCloseable 接口,实现 close() 接口:

  1. @Override
  2. public void close() {
  3. this.stop();
  4. }

修改 start() 方法,使其支持链式调用:

  1. public TraceWatch start(String taskName) throws IllegalStateException {
  2. if (this.currentTaskName != null) {
  3. throw new IllegalStateException("Can't start TraceWatch: it's already running");
  4. }
  5. this.currentTaskName = taskName;
  6. this.startMs = TimeUtils.nowMs();
  7. return this;
  8. }
  9. public class AutoCloseableTest {
  10. public static void main(String[] args) {
  11. TraceWatch traceWatch = new TraceWatch();
  12. try(TraceWatch ignored = traceWatch.start("function1")) {
  13. try {
  14. TimeUnit.SECONDS.sleep(1); // 模拟业务代码
  15. } catch (InterruptedException e) {
  16. e.printStackTrace();
  17. }
  18. }
  19. try(TraceWatch ignored = traceWatch.start("function2")) {
  20. try {
  21. TimeUnit.SECONDS.sleep(1); // 模拟业务代码
  22. } catch (InterruptedException e) {
  23. e.printStackTrace();
  24. }
  25. }
  26. try(TraceWatch ignored = traceWatch.start("function1")) {
  27. try {
  28. TimeUnit.SECONDS.sleep(1); // 模拟业务代码
  29. } catch (InterruptedException e) {
  30. e.printStackTrace();
  31. }
  32. }
  33. System.out.println(JSON.toJSONString(traceWatch.getTaskMap()));
  34. }
  35. }
  36. /* output:
  37. {"function2":[{"data":1001,"taskName":"function2"}],"function1":[{"data":1002,"taskName":"function1"},{"data":1002,"taskName":"function1"}]}
  38. */

总结

本文列举了四种统计代码耗时的方法:

  • 时间差统计
  • StopWatch
  • Function
  • AutoCloseable