siteName > > queue-load-leveling
Threads
java ( 2907718 ) - com.iluwatar.qu ( 2908854 ) stack: com.iluwatar.queue.load.leveling.App.main(App.java:72) org.codehaus.mojo.exec.ExecJavaMojo.doMain(ExecJavaMojo.java:385) org.codehaus.mojo.exec.ExecJavaMojo.doExec(ExecJavaMojo.java:374) org.codehaus.mojo.exec.ExecJavaMojo.lambda$execute$0(ExecJavaMojo.java:296) java.base/java.lang.Thread.run(Thread.java:840)
/*
 * This project is licensed under the MIT license. Module model-view-viewmodel is using ZK framework licensed under LGPL (see lgpl-3.0.txt).
 *
 * The MIT License
 * Copyright © 2014-2022 Ilkka Seppälä
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in
 * all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 * THE SOFTWARE.
 */
package com.iluwatar.queue.load.leveling;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;
import lombok.extern.slf4j.Slf4j;

/**
 * Many solutions in the cloud involve running tasks that invoke services. In this environment, if a
 * service is subjected to intermittent heavy loads, it can cause performance or reliability
 * issues.
 *
 * <p>A service could be a component that is part of the same solution as the tasks that utilize
 * it, or it could be a third-party service providing access to frequently used resources such as a
 * cache or a storage service. If the same service is utilized by a number of tasks running
 * concurrently, it can be difficult to predict the volume of requests to which the service might be
 * subjected at any given point in time.
 *
 * <p>We will build a queue-based-load-leveling to solve above problem. Refactor the solution and
 * introduce a queue between the task and the service. The task and the service run asynchronously.
 * The task posts a message containing the data required by the service to a queue. The queue acts
 * as a buffer, storing the message until it is retrieved by the service. The service retrieves the
 * messages from the queue and processes them. Requests from a number of tasks, which can be
 * generated at a highly variable rate, can be passed to the service through the same message
 * queue.
 *
 * <p>The queue effectively decouples the tasks from the service, and the service can handle the
 * messages at its own pace irrespective of the volume of requests from concurrent tasks.
 * Additionally, there is no delay to a task if the service is not available at the time it posts a
 * message to the queue.
 *
 * <p>In this example we have a class {@link MessageQueue} to hold the message {@link Message}
 * objects. All the worker threads {@link TaskGenerator} will submit the messages to the
 * MessageQueue. The service executor class {@link ServiceExecutor} will pick up one task at a time
 * from the Queue and execute them.
 */
@Slf4j
public class App {

  //Executor shut down time limit.
  private static final int SHUTDOWN_TIME = 15;

  /**
   * Program entry point.
   *
   * @param args command line args
   */
  public static void main(String[] args) {

    // An Executor that provides methods to manage termination and methods that can 
    // produce a Future for tracking progress of one or more asynchronous tasks.
    ExecutorService executor = null;

    try {
      // Create a MessageQueue object.
      var msgQueue = new MessageQueue();

      LOGGER.info("Submitting TaskGenerators and ServiceExecutor threads.");

      // Create three TaskGenerator threads. Each of them will submit different number of jobs.
      final var taskRunnable1 = new TaskGenerator(msgQueue, 5);
      final var taskRunnable2 = new TaskGenerator(msgQueue, 1);
      final var taskRunnable3 = new TaskGenerator(msgQueue, 2);

      // Create e service which should process the submitted jobs.
      final var srvRunnable = new ServiceExecutor(msgQueue);

      // Create a ThreadPool of 2 threads and
      // submit all Runnable task for execution to executor..
      executor = Executors.newFixedThreadPool(2);
      executor.submit(taskRunnable1);
      executor.submit(taskRunnable2);
      executor.submit(taskRunnable3);

      // submitting serviceExecutor thread to the Executor service.
      executor.submit(srvRunnable);

      // Initiates an orderly shutdown.
      LOGGER.info("Initiating shutdown."
          + " Executor will shutdown only after all the Threads are completed.");
      executor.shutdown();

      // Wait for SHUTDOWN_TIME seconds for all the threads to complete 
      // their tasks and then shut down the executor and then exit. 
      if (!executor.awaitTermination(SHUTDOWN_TIME, TimeUnit.SECONDS)) {
        LOGGER.info("Executor was shut down and Exiting.");
        executor.shutdownNow();
      }
    } catch (Exception e) {
      LOGGER.error(e.getMessage());
    }
  }
}
Variables All
No.FromNameValue
1class@72LOGGERLogger[com.iluwatar.queue.load.leveling.App]
2class@72SHUTDOWN_TIME15
372args[Ljava.lang.String;@1430daa8
END 0 00
Output All Filter Merge
Process FilterThread Filter
2907718 java 2908994 pool-1-thread-2 2908854 com.iluwatar.qu 2908990 pool-1-thread-1
No.PNPIDTIDTNTLMessage
1java29077182908854com.iluwatar.qucom.iluwatar.queue.load.leveling.AppISubmitting TaskGenerators and ServiceExecutor threads.
2java29077182908854com.iluwatar.qucom.iluwatar.queue.load.leveling.AppIInitiating shutdown. Executor will shutdown only after all the Threads are completed.
3java29077182908990pool-1-thread-1com.iluwatar.queue.load.leveling.TaskGeneratorIMessage-5 submitted by pool-1-thread-1
4java29077182908994pool-1-thread-2com.iluwatar.queue.load.leveling.TaskGeneratorIMessage-1 submitted by pool-1-thread-2
5java29077182908990pool-1-thread-1com.iluwatar.queue.load.leveling.TaskGeneratorIMessage-4 submitted by pool-1-thread-1
6java29077182908994pool-1-thread-2com.iluwatar.queue.load.leveling.TaskGeneratorIMessage-2 submitted by pool-1-thread-2
7java29077182908990pool-1-thread-1com.iluwatar.queue.load.leveling.TaskGeneratorIMessage-3 submitted by pool-1-thread-1
8java29077182908994pool-1-thread-2com.iluwatar.queue.load.leveling.TaskGeneratorIMessage-1 submitted by pool-1-thread-2
9java29077182908994pool-1-thread-2com.iluwatar.queue.load.leveling.ServiceExecutorIMessage-5 submitted by pool-1-thread-1 is served.
10java29077182908990pool-1-thread-1com.iluwatar.queue.load.leveling.TaskGeneratorIMessage-2 submitted by pool-1-thread-1
11java29077182908994pool-1-thread-2com.iluwatar.queue.load.leveling.ServiceExecutorIMessage-1 submitted by pool-1-thread-2 is served.
12java29077182908990pool-1-thread-1com.iluwatar.queue.load.leveling.TaskGeneratorIMessage-1 submitted by pool-1-thread-1
13java29077182908994pool-1-thread-2com.iluwatar.queue.load.leveling.ServiceExecutorIMessage-4 submitted by pool-1-thread-1 is served.
14java29077182908994pool-1-thread-2com.iluwatar.queue.load.leveling.ServiceExecutorIMessage-2 submitted by pool-1-thread-2 is served.
15java29077182908994pool-1-thread-2com.iluwatar.queue.load.leveling.ServiceExecutorIMessage-3 submitted by pool-1-thread-1 is served.
16java29077182908994pool-1-thread-2com.iluwatar.queue.load.leveling.ServiceExecutorIMessage-1 submitted by pool-1-thread-2 is served.
17java29077182908994pool-1-thread-2com.iluwatar.queue.load.leveling.ServiceExecutorIMessage-2 submitted by pool-1-thread-1 is served.
18java29077182908994pool-1-thread-2com.iluwatar.queue.load.leveling.ServiceExecutorIMessage-1 submitted by pool-1-thread-1 is served.
19java29077182908994pool-1-thread-2com.iluwatar.queue.load.leveling.ServiceExecutorIService Executor: Waiting for Messages to serve ..
20java29077182908994pool-1-thread-2com.iluwatar.queue.load.leveling.ServiceExecutorIService Executor: Waiting for Messages to serve ..
21java29077182908994pool-1-thread-2com.iluwatar.queue.load.leveling.ServiceExecutorIService Executor: Waiting for Messages to serve ..
22java29077182908994pool-1-thread-2com.iluwatar.queue.load.leveling.ServiceExecutorIService Executor: Waiting for Messages to serve ..
23java29077182908854com.iluwatar.qucom.iluwatar.queue.load.leveling.AppIExecutor was shut down and Exiting.
24java29077182908994pool-1-thread-2com.iluwatar.queue.load.leveling.ServiceExecutorEsleep interrupted
END 0 0 0 0 0 00
Project:JavaDesignPatterns
Update:20240509
Commit:bf6456ba6
Source Code:queue-load-leveling
BuildTool:Java17
Compiler:Java17
Runtime:Openjdk17
System:MySystemD
Kernel:Linux5.10.211
Cpu:Intel:Corei7-7700K
Machine:AwesomeMachine