All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.github.rholder.moar.concurrent.HeapQueueingStrategy Maven / Gradle / Ivy

Go to download

This module contains a collection of useful builders and concurrency classes to assist in modeling complex or overly tweakable concurrent processing pipelines.

There is a newer version: 1.0.3
Show newest version
package com.github.rholder.moar.concurrent;

/**
 * This QueueingStrategy slows down the rate at which items can be added to a
 * queue based on the amount of free heap space available.
 *
 * @author rholder
 * @param  the type of elements held in the target queue
 */
public class HeapQueueingStrategy implements QueueingStrategy {

    private static final Runtime RUNTIME = Runtime.getRuntime();

    private long minimumHeapSpaceBeforeFlowControl;
    private long dequeueHint;
    private long dequeued;
    private double c;

    /**
     * Construct a new {@link HeapQueueingStrategy} with the given parameters.
     *
     * @param percentOfHeapBeforeFlowControl the percentage of the heap that must be available before the
     *                                       queue begins to start delaying addition operations
     * @param maxDelay                       the maximum amount of time to delay an addition operation in
     *                                       milliseconds
     * @param dequeueHint                    after this many dequeue operations, signal the JVM to run a
     *                                       garbage collection
     */
    public HeapQueueingStrategy(double percentOfHeapBeforeFlowControl, long maxDelay, long dequeueHint) {
        // convert percentage to the actual heap threshold
        if (percentOfHeapBeforeFlowControl > 0.0 || percentOfHeapBeforeFlowControl <= 1.0) {
            minimumHeapSpaceBeforeFlowControl = Math.round(RUNTIME.maxMemory() * percentOfHeapBeforeFlowControl);
        } else {
            throw new RuntimeException("Range must be between 0.0 and 1.0");
        }

        this.dequeueHint = dequeueHint;
        this.dequeued = 0L; // use this as a hint about when to ask for GC

        // constant used to roll in to y = x^2 * c, this will control the rate at which the queue is staggered
        c = maxDelay / ((double) minimumHeapSpaceBeforeFlowControl * minimumHeapSpaceBeforeFlowControl);
    }

    /**
     * Block for a varying amount based on how close the system is to the max
     * heap space. Only kick in when we have passed the
     * percentOfHeapBeforeFlowControl threshold.
     *
     * @param value value that is to be added to the queue
     */
    public void onBeforeAdd(E value) {
        long freeHeapSpace = RUNTIME.freeMemory() + (RUNTIME.maxMemory() - RUNTIME.totalMemory());

        // start flow control if we cross the threshold
        if (freeHeapSpace < minimumHeapSpaceBeforeFlowControl) {

            // x indicates how close we are to overflowing the heap
            long x = minimumHeapSpaceBeforeFlowControl - freeHeapSpace;

            long delay = Math.round(x * (x * c)); // delay = x^2 * c

            try {
                Thread.sleep(delay);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
        }
    }

    public void onAfterAdd() {
        // do nothing
    }

    public void onBeforeRemove() {
        // do nothing
    }

    /**
     * Increment the count of removed items from the queue. Calling this will
     * optionally request that the garbage collector run to free up heap space
     * for every nth dequeue, where n is the value set for the dequeueHint.
     *
     * @param value value that was removed from the queue
     */
    public void onAfterRemove(E value) {
        if (value != null) {
            dequeued++;
            if (dequeued % dequeueHint == 0) {
                RUNTIME.gc();
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy