StopWatch.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */

  17. package org.apache.commons.lang3.time;

  18. import java.time.Duration;
  19. import java.time.Instant;
  20. import java.util.Objects;
  21. import java.util.concurrent.TimeUnit;

  22. import org.apache.commons.lang3.StringUtils;
  23. import org.apache.commons.lang3.function.FailableConsumer;
  24. import org.apache.commons.lang3.function.FailableRunnable;

  25. /**
  26.  * {@link StopWatch} provides a convenient API for timings.
  27.  *
  28.  * <p>
  29.  * To start the watch, call {@link #start()} or {@link StopWatch#createStarted()}. At this point you can:
  30.  * </p>
  31.  * <ul>
  32.  * <li>{@link #split()} the watch to get the time whilst the watch continues in the background. {@link #unsplit()} will remove the effect of the split. At this
  33.  * point, these three options are available again.</li>
  34.  * <li>{@link #suspend()} the watch to pause it. {@link #resume()} allows the watch to continue. Any time between the suspend and resume will not be counted in
  35.  * the total. At this point, these three options are available again.</li>
  36.  * <li>{@link #stop()} the watch to complete the timing session.</li>
  37.  * </ul>
  38.  *
  39.  * <p>
  40.  * It is intended that the output methods {@link #toString()} and {@link #getTime()} should only be called after stop, split or suspend, however a suitable
  41.  * result will be returned at other points.
  42.  * </p>
  43.  *
  44.  * <p>
  45.  * NOTE: As from v2.1, the methods protect against inappropriate calls. Thus you cannot now call stop before start, resume before suspend or unsplit before
  46.  * split.
  47.  * </p>
  48.  *
  49.  * <ol>
  50.  * <li>{@link #split()}, {@link #suspend()}, or {@link #stop()} cannot be invoked twice</li>
  51.  * <li>{@link #unsplit()} may only be called if the watch has been {@link #split()}</li>
  52.  * <li>{@link #resume()} may only be called if the watch has been {@link #suspend()}</li>
  53.  * <li>{@link #start()} cannot be called twice without calling {@link #reset()}</li>
  54.  * </ol>
  55.  *
  56.  * <p>
  57.  * This class is not thread-safe
  58.  * </p>
  59.  *
  60.  * @see DurationUtils#of(FailableRunnable)
  61.  * @see DurationUtils#of(FailableConsumer)
  62.  *
  63.  * @since 2.0
  64.  */
  65. public class StopWatch {

  66.     /**
  67.      * Enumeration type which indicates the split status of a StopWatch.
  68.      */
  69.     private enum SplitState {
  70.         SPLIT, UNSPLIT
  71.     }

  72.     /**
  73.      * Enumeration type which indicates the status of a StopWatch.
  74.      */
  75.     private enum State {

  76.         RUNNING {
  77.             @Override
  78.             boolean isStarted() {
  79.                 return true;
  80.             }

  81.             @Override
  82.             boolean isStopped() {
  83.                 return false;
  84.             }

  85.             @Override
  86.             boolean isSuspended() {
  87.                 return false;
  88.             }
  89.         },

  90.         STOPPED {
  91.             @Override
  92.             boolean isStarted() {
  93.                 return false;
  94.             }

  95.             @Override
  96.             boolean isStopped() {
  97.                 return true;
  98.             }

  99.             @Override
  100.             boolean isSuspended() {
  101.                 return false;
  102.             }
  103.         },

  104.         SUSPENDED {
  105.             @Override
  106.             boolean isStarted() {
  107.                 return true;
  108.             }

  109.             @Override
  110.             boolean isStopped() {
  111.                 return false;
  112.             }

  113.             @Override
  114.             boolean isSuspended() {
  115.                 return true;
  116.             }
  117.         },

  118.         UNSTARTED {
  119.             @Override
  120.             boolean isStarted() {
  121.                 return false;
  122.             }

  123.             @Override
  124.             boolean isStopped() {
  125.                 return true;
  126.             }

  127.             @Override
  128.             boolean isSuspended() {
  129.                 return false;
  130.             }
  131.         };

  132.         /**
  133.          * Tests whether the StopWatch is started. A suspended StopWatch is also started.
  134.          *
  135.          * @return boolean If the StopWatch is started.
  136.          */
  137.         abstract boolean isStarted();

  138.         /**
  139.          * Tests whether the StopWatch is stopped. A StopWatch which is not yet started and explicitly stopped is considered stopped.
  140.          *
  141.          * @return boolean If the StopWatch is stopped.
  142.          */
  143.         abstract boolean isStopped();

  144.         /**
  145.          * Tests whether the StopWatch is suspended.
  146.          *
  147.          * @return boolean If the StopWatch is suspended.
  148.          */
  149.         abstract boolean isSuspended();
  150.     }

  151.     private static final long NANO_2_MILLIS = 1000000L;

  152.     /**
  153.      * Creates a StopWatch.
  154.      *
  155.      * @return StopWatch a StopWatch.
  156.      *
  157.      * @since 3.10
  158.      */
  159.     public static StopWatch create() {
  160.         return new StopWatch();
  161.     }

  162.     /**
  163.      * Creates and starts a StopWatch.
  164.      *
  165.      * @return StopWatch a started StopWatch.
  166.      *
  167.      * @since 3.5
  168.      */
  169.     public static StopWatch createStarted() {
  170.         final StopWatch sw = new StopWatch();
  171.         sw.start();
  172.         return sw;
  173.     }

  174.     /**
  175.      * A message for string presentation.
  176.      *
  177.      * @since 3.10
  178.      */
  179.     private final String message;

  180.     /**
  181.      * The current running state of the StopWatch.
  182.      */
  183.     private State runningState = State.UNSTARTED;

  184.     /**
  185.      * Whether the StopWatch has a split time recorded.
  186.      */
  187.     private SplitState splitState = SplitState.UNSPLIT;

  188.     /**
  189.      * The start time in nanoseconds.
  190.      *
  191.      * This field can be removed once we move off of Java 8.
  192.      */
  193.     private long startTimeNanos;

  194.     /**
  195.      * The start Instant.
  196.      * <p>
  197.      * nanoTime is only for elapsed time so we need to also store the currentTimeMillis to maintain the old getStartTime API.
  198.      * </p>
  199.      * <p>
  200.      * On Java 8, Instant has millisecond precision, only later versions use nanoseconds.
  201.      * </p>
  202.      */
  203.     private Instant startInstant;

  204.     /**
  205.      * The end Instant.
  206.      * <p>
  207.      * nanoTime is only for elapsed time so we need to also store the currentTimeMillis to maintain the old getStartTime API.
  208.      * </p>
  209.      * <p>
  210.      * On Java 8, Instant has millisecond precision, only later versions use nanoseconds.
  211.      * </p>
  212.      */
  213.     private Instant stopInstant;

  214.     /**
  215.      * The stop time in nanoseconds.
  216.      *
  217.      * This field can be removed once we move off of Java 8.
  218.      */
  219.     private long stopTimeNanos;

  220.     /**
  221.      * Constructs a new instance.
  222.      */
  223.     public StopWatch() {
  224.         this(null);
  225.     }

  226.     /**
  227.      * Constructs a new instance.
  228.      *
  229.      * @param message A message for string presentation.
  230.      * @since 3.10
  231.      */
  232.     public StopWatch(final String message) {
  233.         this.message = message;
  234.     }

  235.     /**
  236.      * Formats the split time with {@link DurationFormatUtils#formatDurationHMS}.
  237.      *
  238.      * @return the split time formatted by {@link DurationFormatUtils#formatDurationHMS}.
  239.      * @since 3.10
  240.      */
  241.     public String formatSplitTime() {
  242.         return DurationFormatUtils.formatDurationHMS(getSplitDuration().toMillis());
  243.     }

  244.     /**
  245.      * Formats the time formatted with {@link DurationFormatUtils#formatDurationHMS}.
  246.      *
  247.      * @return the time formatted by {@link DurationFormatUtils#formatDurationHMS}.
  248.      * @since 3.10
  249.      */
  250.     public String formatTime() {
  251.         return DurationFormatUtils.formatDurationHMS(getTime());
  252.     }

  253.     /**
  254.      * Gets the Duration on the StopWatch.
  255.      *
  256.      * <p>
  257.      * This is either the Duration between the start and the moment this method is called, or the Duration between start and stop.
  258.      * </p>
  259.      *
  260.      * @return the Duration.
  261.      * @since 3.16.0
  262.      */
  263.     public Duration getDuration() {
  264.         return Duration.ofNanos(getNanoTime());
  265.     }

  266.     /**
  267.      * Gets the message for string presentation.
  268.      *
  269.      * @return the message for string presentation.
  270.      * @since 3.10
  271.      */
  272.     public String getMessage() {
  273.         return message;
  274.     }

  275.     /**
  276.      * Gets the <em>elapsed</em> time in nanoseconds.
  277.      *
  278.      * <p>
  279.      * This is either the time between the start and the moment this method is called, or the amount of time between start and stop.
  280.      * </p>
  281.      *
  282.      * @return the <em>elapsed</em> time in nanoseconds.
  283.      * @see System#nanoTime()
  284.      * @since 3.0
  285.      */
  286.     public long getNanoTime() {
  287.         if (runningState == State.STOPPED || runningState == State.SUSPENDED) {
  288.             return stopTimeNanos - startTimeNanos;
  289.         }
  290.         if (runningState == State.UNSTARTED) {
  291.             return 0;
  292.         }
  293.         if (runningState == State.RUNNING) {
  294.             return System.nanoTime() - startTimeNanos;
  295.         }
  296.         throw new IllegalStateException("Illegal running state has occurred.");
  297.     }

  298.     /**
  299.      * Gets the split Duration on the StopWatch.
  300.      *
  301.      * <p>
  302.      * This is the Duration between start and latest split.
  303.      * </p>
  304.      *
  305.      * @return the split Duration
  306.      *
  307.      * @throws IllegalStateException if the StopWatch has not yet been split.
  308.      * @since 3.16.0
  309.      */
  310.     public Duration getSplitDuration() {
  311.         return Duration.ofNanos(getSplitNanoTime());
  312.     }

  313.     /**
  314.      * Gets the split time in nanoseconds.
  315.      *
  316.      * <p>
  317.      * This is the time between start and latest split.
  318.      * </p>
  319.      *
  320.      * @return the split time in nanoseconds
  321.      *
  322.      * @throws IllegalStateException if the StopWatch has not yet been split.
  323.      * @since 3.0
  324.      */
  325.     public long getSplitNanoTime() {
  326.         if (splitState != SplitState.SPLIT) {
  327.             throw new IllegalStateException("Stopwatch must be split to get the split time.");
  328.         }
  329.         return stopTimeNanos - startTimeNanos;
  330.     }

  331.     /**
  332.      * Gets the split time on the StopWatch.
  333.      *
  334.      * <p>
  335.      * This is the time between start and latest split.
  336.      * </p>
  337.      *
  338.      * @return the split time in milliseconds
  339.      *
  340.      * @throws IllegalStateException if the StopWatch has not yet been split.
  341.      * @since 2.1
  342.      * @deprecated Use {@link #getSplitDuration()}.
  343.      */
  344.     @Deprecated
  345.     public long getSplitTime() {
  346.         return nanosToMillis(getSplitNanoTime());
  347.     }

  348.     /**
  349.      * Gets the Instant this StopWatch was started, between the current time and midnight, January 1, 1970 UTC.
  350.      *
  351.      * @return the Instant this StopWatch was started, between the current time and midnight, January 1, 1970 UTC.
  352.      * @throws IllegalStateException if this StopWatch has not been started
  353.      * @since 3.16.0
  354.      */
  355.     public Instant getStartInstant() {
  356.         return Instant.ofEpochMilli(getStartTime());
  357.     }

  358.     /**
  359.      * Gets the time this StopWatch was started in milliseconds, between the current time and midnight, January 1, 1970 UTC.
  360.      *
  361.      * @return the time this StopWatch was started in milliseconds, between the current time and midnight, January 1, 1970 UTC.
  362.      * @throws IllegalStateException if this StopWatch has not been started
  363.      * @since 2.4
  364.      * @deprecated Use {@link #getStartInstant()}.
  365.      */
  366.     @Deprecated
  367.     public long getStartTime() {
  368.         if (runningState == State.UNSTARTED) {
  369.             throw new IllegalStateException("Stopwatch has not been started");
  370.         }
  371.         // stopTimeNanos stores System.nanoTime() for elapsed time
  372.         return startInstant.toEpochMilli();
  373.     }

  374.     /**
  375.      * Gets the Instant this StopWatch was stopped, between the current time and midnight, January 1, 1970 UTC.
  376.      *
  377.      * @return the Instant this StopWatch was stopped in milliseconds, between the current time and midnight, January 1, 1970 UTC.
  378.      * @throws IllegalStateException if this StopWatch has not been started
  379.      * @since 3.16.0
  380.      */
  381.     public Instant getStopInstant() {
  382.         return Instant.ofEpochMilli(getStopTime());
  383.     }

  384.     /**
  385.      * Gets the time this StopWatch was stopped in milliseconds, between the current time and midnight, January 1, 1970 UTC.
  386.      *
  387.      * @return the time this StopWatch was stopped in milliseconds, between the current time and midnight, January 1, 1970 UTC.
  388.      * @throws IllegalStateException if this StopWatch has not been started
  389.      * @since 3.12.0
  390.      * @deprecated Use {@link #getStopInstant()}.
  391.      */
  392.     @Deprecated
  393.     public long getStopTime() {
  394.         if (runningState == State.UNSTARTED) {
  395.             throw new IllegalStateException("Stopwatch has not been started");
  396.         }
  397.         // stopTimeNanos stores System.nanoTime() for elapsed time
  398.         return stopInstant.toEpochMilli();
  399.     }

  400.     /**
  401.      * Gets the time on the StopWatch.
  402.      *
  403.      * <p>
  404.      * This is either the time between the start and the moment this method is called, or the amount of time between start and stop.
  405.      * </p>
  406.      *
  407.      * @return the time in milliseconds
  408.      * @deprecated Use {@link #getDuration()}.
  409.      */
  410.     @Deprecated
  411.     public long getTime() {
  412.         return nanosToMillis(getNanoTime());
  413.     }

  414.     /**
  415.      * Gets the time in the specified TimeUnit.
  416.      *
  417.      * <p>
  418.      * This is either the time between the start and the moment this method is called, or the amount of time between start and stop. The resulting time will be
  419.      * expressed in the desired TimeUnit with any remainder rounded down. For example, if the specified unit is {@code TimeUnit.HOURS} and the StopWatch time is
  420.      * 59 minutes, then the result returned will be {@code 0}.
  421.      * </p>
  422.      *
  423.      * @param timeUnit the unit of time, not null
  424.      * @return the time in the specified TimeUnit, rounded down
  425.      * @since 3.5
  426.      */
  427.     public long getTime(final TimeUnit timeUnit) {
  428.         return timeUnit.convert(getNanoTime(), TimeUnit.NANOSECONDS);
  429.     }

  430.     /**
  431.      * Tests whether the StopWatch is started. A suspended StopWatch is also started watch.
  432.      *
  433.      * @return boolean If the StopWatch is started.
  434.      * @since 3.2
  435.      */
  436.     public boolean isStarted() {
  437.         return runningState.isStarted();
  438.     }

  439.     /**
  440.      * Tests whether StopWatch is stopped. The StopWatch which's not yet started and explicitly stopped StopWatch is considered as stopped.
  441.      *
  442.      * @return boolean If the StopWatch is stopped.
  443.      * @since 3.2
  444.      */
  445.     public boolean isStopped() {
  446.         return runningState.isStopped();
  447.     }

  448.     /**
  449.      * Tests whether the StopWatch is suspended.
  450.      *
  451.      * @return boolean If the StopWatch is suspended.
  452.      * @since 3.2
  453.      */
  454.     public boolean isSuspended() {
  455.         return runningState.isSuspended();
  456.     }

  457.     /**
  458.      * Converts nanoseconds to milliseconds.
  459.      *
  460.      * @param nanos nanoseconds to convert.
  461.      * @return milliseconds conversion result.
  462.      */
  463.     private long nanosToMillis(final long nanos) {
  464.         return nanos / NANO_2_MILLIS;
  465.     }

  466.     /**
  467.      * Resets the StopWatch. Stops it if need be.
  468.      *
  469.      * <p>
  470.      * This method clears the internal values to allow the object to be reused.
  471.      * </p>
  472.      */
  473.     public void reset() {
  474.         runningState = State.UNSTARTED;
  475.         splitState = SplitState.UNSPLIT;
  476.     }

  477.     /**
  478.      * Resumes the StopWatch after a suspend.
  479.      *
  480.      * <p>
  481.      * This method resumes the watch after it was suspended. The watch will not include time between the suspend and resume calls in the total time.
  482.      * </p>
  483.      *
  484.      * @throws IllegalStateException if the StopWatch has not been suspended.
  485.      */
  486.     public void resume() {
  487.         if (runningState != State.SUSPENDED) {
  488.             throw new IllegalStateException("Stopwatch must be suspended to resume. ");
  489.         }
  490.         startTimeNanos += System.nanoTime() - stopTimeNanos;
  491.         runningState = State.RUNNING;
  492.     }

  493.     /**
  494.      * Splits the time.
  495.      *
  496.      * <p>
  497.      * This method sets the stop time of the watch to allow a time to be extracted. The start time is unaffected, enabling {@link #unsplit()} to continue the
  498.      * timing from the original start point.
  499.      * </p>
  500.      *
  501.      * @throws IllegalStateException if the StopWatch is not running.
  502.      */
  503.     public void split() {
  504.         if (runningState != State.RUNNING) {
  505.             throw new IllegalStateException("Stopwatch is not running. ");
  506.         }
  507.         stopTimeNanos = System.nanoTime();
  508.         splitState = SplitState.SPLIT;
  509.     }

  510.     /**
  511.      * Starts the StopWatch.
  512.      *
  513.      * <p>
  514.      * This method starts a new timing session, clearing any previous values.
  515.      * </p>
  516.      *
  517.      * @throws IllegalStateException if the StopWatch is already running.
  518.      */
  519.     public void start() {
  520.         if (runningState == State.STOPPED) {
  521.             throw new IllegalStateException("Stopwatch must be reset before being restarted. ");
  522.         }
  523.         if (runningState != State.UNSTARTED) {
  524.             throw new IllegalStateException("Stopwatch already started. ");
  525.         }
  526.         startTimeNanos = System.nanoTime();
  527.         startInstant = Instant.now();
  528.         runningState = State.RUNNING;
  529.     }

  530.     /**
  531.      * Stops the StopWatch.
  532.      *
  533.      * <p>
  534.      * This method ends a new timing session, allowing the time to be retrieved.
  535.      * </p>
  536.      *
  537.      * @throws IllegalStateException if the StopWatch is not running.
  538.      */
  539.     public void stop() {
  540.         if (runningState != State.RUNNING && runningState != State.SUSPENDED) {
  541.             throw new IllegalStateException("Stopwatch is not running. ");
  542.         }
  543.         if (runningState == State.RUNNING) {
  544.             stopTimeNanos = System.nanoTime();
  545.             stopInstant = Instant.now();
  546.         }
  547.         runningState = State.STOPPED;
  548.     }

  549.     /**
  550.      * Suspends the StopWatch for later resumption.
  551.      *
  552.      * <p>
  553.      * This method suspends the watch until it is resumed. The watch will not include time between the suspend and resume calls in the total time.
  554.      * </p>
  555.      *
  556.      * @throws IllegalStateException if the StopWatch is not currently running.
  557.      */
  558.     public void suspend() {
  559.         if (runningState != State.RUNNING) {
  560.             throw new IllegalStateException("Stopwatch must be running to suspend. ");
  561.         }
  562.         stopTimeNanos = System.nanoTime();
  563.         stopInstant = Instant.now();
  564.         runningState = State.SUSPENDED;
  565.     }

  566.     /**
  567.      * Gets a summary of the split time that the StopWatch recorded as a string.
  568.      *
  569.      * <p>
  570.      * The format used is ISO 8601-like, [<i>message</i> ]<i>hours</i>:<i>minutes</i>:<i>seconds</i>.<i>milliseconds</i>.
  571.      * </p>
  572.      *
  573.      * @return the split time as a String
  574.      * @since 2.1
  575.      * @since 3.10 Returns the prefix {@code "message "} if the message is set.
  576.      */
  577.     public String toSplitString() {
  578.         final String msgStr = Objects.toString(message, StringUtils.EMPTY);
  579.         final String formattedTime = formatSplitTime();
  580.         return msgStr.isEmpty() ? formattedTime : msgStr + StringUtils.SPACE + formattedTime;
  581.     }

  582.     /**
  583.      * Gets a summary of the time that the StopWatch recorded as a string.
  584.      *
  585.      * <p>
  586.      * The format used is ISO 8601-like, [<i>message</i> ]<i>hours</i>:<i>minutes</i>:<i>seconds</i>.<i>milliseconds</i>.
  587.      * </p>
  588.      *
  589.      * @return the time as a String
  590.      * @since 3.10 Returns the prefix {@code "message "} if the message is set.
  591.      */
  592.     @Override
  593.     public String toString() {
  594.         final String msgStr = Objects.toString(message, StringUtils.EMPTY);
  595.         final String formattedTime = formatTime();
  596.         return msgStr.isEmpty() ? formattedTime : msgStr + StringUtils.SPACE + formattedTime;
  597.     }

  598.     /**
  599.      * Removes a split.
  600.      *
  601.      * <p>
  602.      * This method clears the stop time. The start time is unaffected, enabling timing from the original start point to continue.
  603.      * </p>
  604.      *
  605.      * @throws IllegalStateException if the StopWatch has not been split.
  606.      */
  607.     public void unsplit() {
  608.         if (splitState != SplitState.SPLIT) {
  609.             throw new IllegalStateException("Stopwatch has not been split. ");
  610.         }
  611.         splitState = SplitState.UNSPLIT;
  612.     }

  613. }