Small knowledge, big challenge! This article is participating in the creation activity of “Essential Tips for Programmers”

1. Thread wait implementation

Create a Thread, run it through a while loop, and implement the timing effect through the sleep method.

Simple and direct, but the implementation is limited, the loop is delayed execution.

public class Task {
    public static void main(String[] args) {
        Runnable runnable = new Runnable() {
            @Override
            public void run(a) {
                while (true) {
                    System.out.println("Hello Java!");
                    try {
                        Thread.sleep(2000);
                    } catch(InterruptedException e) { e.printStackTrace(); }}}}; Thread thread =newThread(runnable); thread.start(); }}Copy the code

2. JDK comes with Timer implementation

Timer is a Timer tool that is used to execute a specified task on a background thread according to a schedule, which can be one or more times.

  • The Timer API is an early development in the JDK, injava.util.TimerTaskIn the package
  • You need to specify that the class inherits TimerTaskClass to override the` run () method

2.1 Methods related to Timer

The Timer class is a class in the java.util package that performs scheduled tasks using a queue to store tasks that need to be executed and to start a separate thread to execute them

// Executes the specified task after the specified delay
schedule(TimerTask task,long delay);

// Executes the specified task at the specified time. (Execute only once)
schedule(TimerTask task, Date time);

After the specified delay, the task starts to repeat at the specified period
schedule(TimerTask task,long delay,long period);

// Starts repeating the specified task at the specified time and at the specified period
schedule(TimerTask task, Date firstTime , long period);

// Starts periodic execution at the specified time
scheduleAtFixedRate(TimerTask task,Date firstTime,long period);

// Periodic execution starts after a specified delay
scheduleAtFixedRate(TimerTask task,long delay,long period);

// Terminates the timer task and discards all currently scheduled tasks.Cancal ();// Remove all cancelled tasks from the task queue of this timer.Purge ();Copy the code

2.2 Defining scheduled Task Classes

The scheduled task class needs to inherit the TimerTask class from the java.util package and implement its run() method.

public class DoTask extends TimerTask {

    @Override
    public void run(a) {
        System.out.println("TimerTask execution ----- :"+ System.currentTimeMillis()); }}Copy the code

2.3 Example

To actually use the scheduled task, you need to create an object of the Timer class, call the methods related to the object, and pass in the scheduled task and task triggering mechanism.

Public static void main(String[] args){Timer Timer = new Timer(); The timer. The schedule (new DoTask (), 1000100); }Copy the code

2.4 Timer problem

Timer can perform scheduled tasks, delayed tasks, and periodic tasks. However, Timer also has some problems:

  1. Tasks are performed based on absolute time rather than relative time and can become overly dependent on system time
  2. If an uncaught exception occurs during the execution of a task, all tasks are stopped

Using a TImer in IDEA prompts us to use ScheduledExecutorService instead of a TImer.

3. ScheduledExecutorService

ScheduledExecutorService is a scheduled task interface added in Java1.5. Based on the thread pool mode, each task starts a new thread for execution without affecting each other.

ScheduledExecutorService exists in the java.util.concurrent package

For the problem of using Timer, IDEA also provides us with the method of using the new scheme:

/ / ScheduledExecutorService instead of the Timer
//Inspection Info: When a Timer runs multiple timeTasks, if one of them does not catch the thrown exception, the other tasks will automatically terminate. ScheduledExecutorService does not have this problem.
            
//org.apache.commons.lang3.concurrent.BasicThreadFactory
ScheduledExecutorService executorService = new ScheduledThreadPoolExecutor(1.new BasicThreadFactory.Builder().namingPattern("example-schedule-pool-%d").daemon(true).build());
executorService.scheduleAtFixedRate(new Runnable() {
    @Override
    public void run(a) {
        //do something
    }
},initialDelay,period, TimeUnit.HOURS);
Copy the code

3.1 Methods

ScheduledExecutorService provides the following four methods:

// Create a scheduled task, specify the trigger delay time, the task passed RunnableScheduledFuture<? > schedule(Runnable command,long delay, TimeUnit unit);

// Create a scheduled task, specify the trigger delay time, the task passed a Callable back call
<V> ScheduledFuture<V> schedule(Callable<V> callable,long delay, TimeUnit unit);

InitialDelay is the delay period of the first execution, and period is the interval between two successful execution
// If the program execution time exceeds the interval, the next execution will be executed immediately after the last executionScheduledFuture<? > scheduleAtFixedRate(Runnable command,long initialDelay,long period,TimeUnitunit);

InitialDelay Delay of the first execution, duration between the last end of dalay and the next startScheduledFuture<? > scheduleWithFixedDelay(Runnable command,long initialDelay,long delay,TimeUnitunit);
Copy the code

ScheduledExecutorService, as an interface, needs to use its implementation class when creating its objects and calling methods:

  • ScheduledThreadPoolExecutor class that is used to create ScheduledExecutorService type object
    • Because it also inherits the thread pool creation class, you need to specify the number of core threads when you create it
  • DelegatedScheduledExecutorService, is the Executors of inner classes

3.2 Task Creation

ScheduledExecutorService provides methods that require tasks such as Runnable or Callable

. In fact, the TimerTask class in java.util implements the Runnable interface.

We can implement the Runnable interface by defining the task class directly and passing it as a parameter to ScheduledExecutorService.

public class DoTaskBySchedule implements Runnable {

    @Override
    public void run(a) {
        System.out.println("Task execution, current time:"+ System.currentTimeMillis()); }}Copy the code

3.3 Example

To use ScheduledExecutorService, create an object of type ScheduledExecutorService and execute its methods.

// The main method of the class performs scheduled tasks
public static void main(String[] args){
    ScheduledExecutorService scheduledExecutorService = new ScheduledThreadPoolExecutor(5);
    scheduledExecutorService.scheduleAtFixedRate(new DoTaskBySchedule(),1000.2000, TimeUnit.MILLISECONDS);
}
Copy the code

3.4 Deficiencies

ScheduledExecutorService still has some disadvantages when it is used, such as:

  • There is no execution method that can be specified by time, and you need to pass in the delay time parameter to execute
  • If you specify a scheduled task, the task will not be executed next time if the task is not executed last time

4. To summarize

To sum up, the use method of some timers in Java JDK is ok for simple delayed tasks, but it is not good for complex, delicate and periodic scheduled tasks. For a large number of scheduled tasks used in project business, special framework needs to be introduced to complete them.