/MultiThreading

Repository for classroom codes and reading material for the concept of Multi-Threading in Java

Primary LanguageJava

Multi-Threading in Java

  • A multithreaded program in Java contains two or more parts that can run concurrently, each part of the program is called a thread.
  • Each thread defines a separate path of execution, hence we can say that multithreading is a special case of multitasking.

Type of Multitasking

  1. Process-based
  2. Thread-based
  • A process is essentially a program that is executing, hence, process-based multitasking is the feature that allows a computer to run (or execute) two or more program concurrently. For example, Process-based multitasking allows us to run the Java IDE, Zoom, Google Chrome, WhatsApp etc at the same time on our computer. Here, a program is the smallest unit of code that can be dispatched by the scheduler.

  • In a thread-based multitasking environment, a thread is the smalled unit of code that can be dispatched by the scheduler. This means that a single program can execute two or more tasks simultaneously. For example, we can still edit the code in our IDE when the compiler is compiling it or the JVM is executing it, as long as these two tasks are being performed on two different threads.

Thus, we can say that,

  • The process-based multitasking handles the "big-picture", and
  • The thread-based multitasking handles the "fine-details".

Multitasking threads require less resources than multitasking processes.

The Java Thread Model

The multi-threading capabilities of Java are built-in.

States of a thread in Java

  1. Running (currently executing)
  2. Suspended (paused, after running)
  3. Resumed (starting after paused)
  4. Blocked (not running, and waiting for a resource)
  5. Ready (waiting to run)
  6. Terminated (halted OR stopped)

Thread Priorities The Java compiler will assign a priority to a thread. This priority will tell us that how which thread should be treated with respect to others. Basically, these are some integer values. But a higher priority of a thread can not guarantee that a thread will perform its task fast or slow, if it is the only thread running. Priority of a thread can only be observed properly once there are two or more threads running concurrently in a program space. It is basically used to decide that when to switch from one running thread to the next (context switch). There are basic two rules for context switching

  1. A thread can voluntarily relinquish(to lose) control: This happens when explicitly yielding, sleeping, or when the thread is blocked. In this, all other threads are examines, and the thread with the highest priority will be marked as ready to run.
  2. A thread can be preempted by a higher-priority thread: In this, a thread with a low-priority does not yield the processor, but when it is preempted by a higher-priority processor, it will yield.

The Thread class and the Runnable Interface.

Java's multithreading system is built upon the Thread class, and its companion interface, the Runnable interface.

The Main Thread

  • Whenever a Java program executes, one thread starts running immediately, this is called the main thread of the program.

  • It is important for two main reasons.

  1. It is the thread from which all the other threads will be created.
  2. It must be the last thread to finish execution because it performs various shutdown actions.

Creating a Thread

  1. By implementing the Runnable interface.
  2. By extending the Thread class.

1. By implementing the Runnable Interface

  • Create a definition class.
class ThreadWithRunnableInterface {
}
  • Implement the Runnable interface in the class.
class ThreadWithRunnableInterface implements Runnable {
}
  • Implement the run() method from the Runnable Interface; remember, the java.lang.Runnable interface is a functional interface.
class ThreadWithRunnable implements Runnable {
    @Override
    public void run() {}
}
  • Create a field of the Thread type for the class; make it private and final.
class ThreadWithRunnable implements Runnable {
    private final Thread thread;
    @Override
    public void run() {}
}
  • Create a public constructor for the class.
class ThreadWithRunnable implements Runnable {
    private final Thread thread;

    public ThreadWithRunnable() {}    

    @Override
    public void run() {}
}
  • Initialize the thread field inside the constructor.

The String parameter inside the constructor of the Thread class is the name of the thread.

class ThreadWithRunnable implements Runnable {
    private final Thread thread;

    public ThreadWithRunnable() {
        thread = new Thread(this, "Thread Name");
    }    

    @Override
    public void run() {}
}
  • Optional: If you would like to expose the thread field outside this class, you may do so by defining a getter method for the field.
class ThreadWithRunnable implements Runnable {
    private final Thread thread;

    public ThreadWithRunnable() {
        thread = new Thread(this, "Thread Name");
    }    

    public Thread getThread() {
        return thread;
    }

    @Override
    public void run() {}
}

AND THAT'S ABOUT IT!! Now, you can create an object of this class and call the start() method to execute this thread.

class ThreadRunner {
    public static void main(String[] args){
        ThreadWithRunnable thread1 = new ThreadWithRunnable();
        thread1.start();
    }
}

EXTRA: You can specify your thread to sleep(pause) for some time using the static sleep() method of the java.lang.Thread class.
Just remember that this method might cause an InterruptedException, since this is an unhandled exception, we will wrap the sleep() method inside a try/catch block.

class ThreadWithRunnable implements Runnable {
    private final Thread thread;

    public ThreadWithRunnable() {
        thread = new Thread(this, "Thread Name");
    }    

    @Override
    public void run() {
        try {
            Thread.sleep(500L);
        } catch (InterruptedException e) {
            System.err.println(e.getMessage());
        }
    }
}

2. By extending the Thread class.

  • Create a definition class.
class ThreadWithThreadClass {
}
  • Extend the java.lang.Thread class; since the Thread class is present in the java.lang package, we do not need to import it specifically.
class ThreadWithThreadClass extends Thread {
}
  • Create a constructor and call the constructor of the parent class (The Thread class) via super(). In the call to the super() constructor, pass the name of the thread.
class ThreadWithThreadClass extends Thread {

    public ThreadWithThreadClass() {
        super("SimpleThread");
    }

}
  • EXTRA If you want, you can overload the constructor also like this
class ThreadWithThreadClass extends Thread {

    public ThreadWithThreadClass() {
        super("SimpleThread");
    }

    public ThreadWithThreadClass(String threadName){
        super(threadName);
    }

}
  • Next, we need to override the run() method from the Thread class.
class ThreadWithThreadClass extends Thread {

    public ThreadWithThreadClass() {
        super("SimpleThread");
    }

    @Override
    public void run() {
        // Specify the task for your thread here!
    }   

}

AND THAT'S ABOUT IT!! Now, you can create an object of this class and call the run() method to execute this thread.

class ThreadRunner {
    public static void main(String[] args){
        ThreadWithThreadClass thread1 = new ThreadWithThreadClass("PUBG");
        thread1.run();
    }
}