org.javasimon.Split Maven / Gradle / Ivy
package org.javasimon;
import java.util.Iterator;
import java.util.Map;
import org.javasimon.clock.SimonClock;
import org.javasimon.utils.SimonUtils;
/**
* Represents single time split - one Stopwatch measurement. Object is obtained by {@link org.javasimon.Stopwatch#start()}
* and the measurement is ended using {@link #stop()} method on this object. Split will return 0 as the result
* if the related Stopwatch was disabled when the Split was obtained. The Split can be stopped in any other thread.
* Split measures real time (based on {@link org.javasimon.clock.SimonClock#nanoTime()}), it does not measure CPU time. Split can be garbage collected
* and no resource leak occurs if it is not stopped, however Stopwatch's active counter ({@link org.javasimon.Stopwatch#getActive()})
* will be stay incremented.
*
* Split can never be running ({@link #isRunning()}) if it is disabled. Enabled split is running until it is stopped.
* Stopped split (not running) will never again be running. Split never changes enabled flag after creation.
*
* Split implements {@link java.lang.AutoCloseable} hence it can be used in try-with-resource construction.
*
* @author Richard "Virgo" Richter
* @see Stopwatch
*/
public final class Split implements HasAttributes, AutoCloseable {
/** Disabled split (implies not running) for cases where monitoring is disabled and {@code null} value is not an option. */
public static final Split DISABLED = new Split();
/** Attribute name under which effectively used stopwatch is stored if the split was stopped with {@link #stop(String)}. */
public static final String ATTR_EFFECTIVE_STOPWATCH = "effective-stopwatch";
private volatile Stopwatch stopwatch;
private final boolean enabled;
private final SimonClock clock;
private volatile boolean running;
private volatile long start;
private volatile long total;
private AttributesSupport attributesSupport = new AttributesSupport();
private Split() {
enabled = false;
clock = null;
}
private Split(boolean enabled, SimonClock clock) {
this.enabled = enabled;
this.clock = clock;
start = clock.nanoTime();
}
/**
* Creates a new Split for an enabled Stopwatch with a specific timestamp in nanoseconds - called internally only.
*
* @param stopwatch owning Stopwatch (enabled)
* @param clock Clock for this Split
* @param start start timestamp in nanoseconds
*/
Split(Stopwatch stopwatch, SimonClock clock, long start) {
assert start > 0 : "start ns value should not be 0 in this constructor!";
this.stopwatch = stopwatch;
this.start = start;
this.clock = clock;
enabled = true;
running = true;
}
/**
* Creates a new Split for a disabled Stopwatch - called internally only.
*
* @param stopwatch owning Stopwatch (disabled)
* @param clock Clock for this Split
*/
Split(Stopwatch stopwatch, SimonClock clock) {
assert !(stopwatch.isEnabled()) : "stopwatch must be disabled in this constructor!";
this.enabled = false;
this.stopwatch = stopwatch;
this.clock = clock;
}
/**
* Creates a new Split for direct use without {@link Stopwatch} ("anonymous split") based on specified {@link SimonClock}.
* Stop will not update any Stopwatch, value can be added to any chosen Stopwatch using
* {@link Stopwatch#addSplit(Split)} - even in conjunction with {@link #stop()} like this:
*
* Split split = Split.start();
* ...
* stopwatch.addSplit(split.stop());
*
* If the split is not needed afterwards (subject to garbage collection) calling {@link #stop()} is not necessary:
*
* Split split = Split.start();
* ...
* stopwatch.addSplit(split);
*
* No callbacks are called for this Split.
*
* @param clock Clock for this Split
* @since 3.5
*/
public static Split start(SimonClock clock) {
Split split = new Split(true, clock);
split.running = true;
return split;
}
/**
* Creates a new Split for direct use without {@link Stopwatch} ("anonymous split") based on system time.
* Equivalent of {@code Split.start(Clock.SYSTEM)}.
*
* @see #start(SimonClock)
* @since 3.4
*/
public static Split start() {
return start(SimonClock.SYSTEM);
}
/**
* Creates simulated non-running Split that took specific time in nanos.
* No callbacks are called for this Split. Start of this Split is set {@code clock.nanoTime()}.
*
* @param nanos Split's total time in nanos
* @param clock Clock for this Split
* @return created Split
* @since 3.5
*/
public static Split create(long nanos, SimonClock clock) {
Split split = new Split(false, clock);
split.total = nanos;
return split;
}
/**
* Creates simulated non-running Split that took specific time in nanos.
* No callbacks are called for this Split. Start of this Split is based on {@link SimonClock#SYSTEM}.
*
* @param nanos Split's total time in nanos
* @return created Split
* @see #create(long, SimonClock)
* @since 3.5
*/
public static Split create(long nanos) {
return create(nanos, SimonClock.SYSTEM);
}
/**
* Returns the stopwatch that this split is running for. May be {@code null} for anonymous splits (directly created).
*
* @return owning stopwatch, may return {@code null}
*/
public Stopwatch getStopwatch() {
return stopwatch;
}
/**
* Stops the split, updates the stopwatch and returns this. Subsequent calls do not change the state of the split.
*
* @return this split object
* @since 3.1 - previously returned split time in ns, call additional {@link #runningFor()} for the same result
*/
public Split stop() {
return stop(null);
}
/**
* Stops the split, updates the sub-stopwatch specified by parameter and returns this. Active counter of the original stopwatch
* is decreased as expected. Subsequent calls do not change the state of the split.
*
* Important: While {@link #stop()} (or this stop with {@code null} argument)
* results in {@link org.javasimon.callback.Callback#onStopwatchStop(Split, StopwatchSample)} callback method being invoked,
* if sub-simon is affected then {@link org.javasimon.callback.Callback#onStopwatchAdd(Stopwatch, Split, StopwatchSample)}
* is called instead.
*
* If the split was obtained from disabled Stopwatch, this method does not update sub-simon even if it is enabled, because
* split itself is disabled as well. If split is enabled, but sub-simon is disabled, the latter is not updated.
*
* @param subSimon name of the sub-stopwatch (hierarchy delimiter is added automatically) - if {@code null}
* it behaves exactly like {@link #stop()}
* @return this split object
* @since 3.4
*/
public Split stop(String subSimon) {
if (!running) {
return this;
}
running = false;
long nowNanos = clock.nanoTime();
total = nowNanos - start; // we update total before calling the stop so that callbacks can use it
if (stopwatch != null) {
((StopwatchImpl) stopwatch).stop(this, start, nowNanos, subSimon);
}
return this;
}
/**
* Returns the current running nano-time from the start to the method call or the total split time
* if the Split has been stopped already.
*
* @return current running nano-time of the split
*/
public long runningFor() {
if (!running) {
return total;
}
if (enabled) {
return clock.nanoTime() - start;
}
return 0;
}
/**
* Returns printable form of how long this split was running for.
*
* @return short information about the Split time as a human readable string
* @since 2.2
*/
public String presentRunningFor() {
return SimonUtils.presentNanoTime(runningFor());
}
/**
* Returns true if this split was created from enabled Simon or via {@link #start()}.
*
* @return true if this split was created from enabled Simon
*/
public boolean isEnabled() {
return enabled;
}
/**
* Returns true if this split is still running ({@link #stop()} has not been called yet).
* Returns false for disabled Split.
*
* @return true if this split is still running
* @since 3.1.0
*/
public boolean isRunning() {
return running;
}
/**
* Returns start nano timer value or 0 if the Split is not enabled (started for disabled Stopwatch).
*
* @return nano timer value when the Split was started or 0 if the Split is not enabled
* @since 3.1
*/
public long getStart() {
return start;
}
/**
* Returns millis timestamp when this Split was started - or 0 if the Split is not enabled (started for disabled Stopwatch).
*
* @return nano timer value when the Split was started or 0 if the Split is not enabled
* @since 3.1
*/
public long getStartMillis() {
return clock != null ? clock.millisForNano(start) : 0;
}
/**
* Stores an attribute in this Split. Attributes can be used to store any custom objects.
*
* @param name a String specifying the name of the attribute
* @param value the Object to be stored
* @since 2.3
*/
@Override
public void setAttribute(String name, Object value) {
attributesSupport.setAttribute(name, value);
}
/**
* Returns the value of the named attribute as an Object, or {@code null} if no attribute of
* the given name exists.
*
* @param name a String specifying the name of the attribute
* @return an Object containing the value of the attribute, or {@code null} if the attribute does not exist
* @since 2.3
*/
@Override
public Object getAttribute(String name) {
return attributesSupport.getAttribute(name);
}
/**
* Returns the value of the named attribute typed to the specified class, or {@code null} if no attribute of
* the given name exists.
*
* @param name a String specifying the name of the attribute
* @return the value of the attribute typed to T, or {@code null} if the attribute does not exist
* @since 3.4
*/
@SuppressWarnings("unchecked")
@Override
public T getAttribute(String name, Class clazz) {
return attributesSupport.getAttribute(name, clazz);
}
/**
* Removes an attribute from this Split.
*
* @param name a String specifying the name of the attribute to remove
* @since 2.3
*/
@Override
public void removeAttribute(String name) {
attributesSupport.removeAttribute(name);
}
/**
* Returns an Iterator containing the names of the attributes available to this Split.
* This method returns an empty Iterator if the Split has no attributes available to it.
*
* @return an Iterator of strings containing the names of the Split's attributes
* @since 2.3
*/
@Override
public Iterator getAttributeNames() {
return attributesSupport.getAttributeNames();
}
@Override
public Map getCopyAsSortedMap() {
return attributesSupport.getCopyAsSortedMap();
}
/**
* Allows to use Split as a resource in try-with-resource construct. Calls {@link #stop()} internally.
*
* @since 4.0
*/
@Override
public void close() {
stop();
}
/**
* Returns information about this Split, if it's running, name of the related Stopwatch and split's time.
*
* @return information about the Split as a human readable string
*/
@Override
public String toString() {
if (!enabled) {
return "Split created from disabled Stopwatch";
}
if (running) {
return "Running split" + (stopwatch != null ? " for Stopwatch '" + stopwatch.getName() + "': " : ": ") + SimonUtils.presentNanoTime(runningFor());
}
return "Stopped split" + (stopwatch != null ? " for Stopwatch '" + stopwatch.getName() + "': " : ": ") + SimonUtils.presentNanoTime(total);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy