 
            Exploring multi-threading
Hi there! Hope you are doing good. I always had this fear of certain niche topics in software engineering which keep on bothering me every now and then (M kinda noob 🐼). Multi-threading is one such topic.
So hence I have decided to face my inner-fears 💪 and create a series of posts which will help me and others with similar fears to understand these topics in an easier way. This will also serve as reference notes.
Basic Understanding on Threads
- Threads essentially allow us to solve computational problems by dividing the larger problem into smaller parallelized tasks. These tasks can run asynchronously and in a non-blocking fashion. Threads are especially useful in boosting performance ⚡ in multi-core systems.
- Threads are often called as light-weight processes. Creating a new thread has a lesser memory foot-print as compared to creating a new process.
- Every process has atleast has one thread(main thread). Using the main thread we can create child threads as per our needs.
- Threads share resources of the parent process like memory space. Hence, data consistency can be an issue.
Creating a thread in java
- There are mainly two ways in which you can create threads in java:
1. Extending Thread class:
- One way to create a thread is to extend the java.lang.Threadclass and provide an implementation for therun()method by overriding it.
|  |  | 
|  |  | 
- 
By default, the Threadclass provides an empty run method implementation as default implementation. Therun()method is where you need to provide your code which needs to be run as a thread.
- 
However, this is not a recommended approach to create Threads as the SampleThread class now cannot extend any other class if we wanted it to. 
2. Implementing the Runnable interface (Recommended way)
- This is the recommended approach as its more flexible. We are not subclassing a concrete class to add threading functionality. We are implementing the run()method ofjava.lang.Runnableinterface.
|  |  | 
|  |  | 
- 
Notice how we are passing the Runnableobject into the Thread class object.
- 
Also, we are now invoking start()method instead ofrun().
- 
Since Runnableinterface is a Functional interface, we can use lambdas to pass our thread code which makes our code concise and makes you look cool 😎.
|  |  | 
- Thread.sleep(long millis)can be used to pause a thread. It throws- InterruptedException, hence the exception needs to be handled using a try-catch block.
join() method
- The join method allows one thread to wait for the completion of another.
- If tis aThreadobject whose thread is currently executing,t.join();causes the current(parent) thread to pause execution untilt'sthread terminates.
- This is helpful when one thread is dependent on another thread’s completion. We can make the dependent thread wait for its dependencies to execute first.
In the upcoming posts, I will try to cover concurency and synchronization. Have a nice day! 😇