Home >Java >javaTutorial >What are the ways to implement multithreading in Java?
There are four main ways to implement Java multi-threading:
① Inherit the Thread class and implement the Runnable interface
② Implement the Callable interface and create it through the FutureTask wrapper Thread thread
③ Use ExecutorService and Callable
④ Future to implement multi-threading with return results
The first two methods have no return value after the thread is executed, and the latter two methods The kind has a return value.
The Thread class is essentially an instance that implements the Runnable interface, representing a thread. Example. Using the start() instance method of the Thread class is the only way to start a thread. A new thread executing the run() method is started by calling the start() method, which is a native method. It is very simple to implement multi-threading in this way. By directly extending Thread through your own class and overriding the run() method, you can start a new thread and execute your own defined run() method. For example:
public class MyThread extends Thread { public void run() { System.out.println("MyThread.run()"); } } MyThread myThread1 = new MyThread(); MyThread myThread2 = new MyThread(); myThread1.start(); myThread2.start();
If your class has extended another class, you cannot directly extend Thread. At this time, you can implement a Runnable interface, as follows:
public class MyThread extends OtherClass implements Runnable { public void run() { System.out.println("MyThread.run()"); } }
In order to start MyThread, you need to instantiate a Thread first and pass in your own MyThread instance:
MyThread myThread = new MyThread(); Thread thread = new Thread(myThread); thread.start();
In fact, when a Runnable target parameter is passed to Thread, Thread's run( ) method will call target.run(), refer to the JDK source code:
public void run() { if (target != null) { target.run(); } }
Create a Thread thread through the FutureTask wrapper
Callable interface (There is only one method) defined as follows:
public interface Callable<V> { V call() throws Exception; } public class SomeCallable<V> extends OtherClass implements Callable<V> { @Override public V call() throws Exception { // TODO Auto-generated method stub return null; } }
Callable<V> oneCallable = new SomeCallable<V>(); //由Callable<Integer>创建一个FutureTask<Integer>对象: FutureTask<V> oneTask = new FutureTask<V>(oneCallable); //注释:FutureTask<Integer>是一个包装器,它通过接受Callable<Integer>来创建,它同时实现了Future和Runnable接口。 //由FutureTask<Integer>创建一个Thread对象: Thread oneThread = new Thread(oneTask); oneThread.start(); //至此,一个线程就创建完成了。
Use ExecutorService, Callable, and Future to implement a thread that returns a result
ExecutorService, The three interfaces Callable and Future actually belong to the Executor framework. In JDK1.5, threads that return results are introduced as a new feature, so you no longer need to go through trouble to obtain the return value. And even if you implement it yourself, it may be full of loopholes.
Tasks that can return values must implement the Callable interface. Similarly, tasks that do not return a value must implement the Runnable interface.
After executing the Callable task, you can obtain a Future object. By calling get on the object, you can obtain the Object returned by the Callable task.
Note: The get method is blocking, that is: the thread returns no result, and the get method will wait forever.
Combined with the thread pool interface ExecutorService, the legendary multi-threading with returned results can be realized.
It has been verified under JDK1.5 and there is no problem. You can directly use the multi-threaded test example with returned results provided below. The code is as follows:
import java.util.concurrent.*; import java.util.Date; import java.util.List; import java.util.ArrayList; /** * 有返回值的线程 */ @SuppressWarnings("unchecked") public class Test { public static void main(String[] args) throws ExecutionException, InterruptedException { System.out.println("----程序开始运行----"); Date date1 = new Date(); int taskSize = 5; // 创建一个线程池 ExecutorService pool = Executors.newFixedThreadPool(taskSize); // 创建多个有返回值的任务 List<Future> list = new ArrayList<Future>(); for (int i = 0; i < taskSize; i++) { Callable c = new MyCallable(i + " "); // 执行任务并获取Future对象 Future f = pool.submit(c); // System.out.println(">>>" + f.get().toString()); list.add(f); } // 关闭线程池 pool.shutdown(); // 获取所有并发任务的运行结果 for (Future f : list) { // 从Future对象上获取任务的返回值,并输出到控制台 System.out.println(">>>" + f.get().toString()); } Date date2 = new Date(); System.out.println("----程序结束运行----,程序运行时间【" + (date2.getTime() - date1.getTime()) + "毫秒】"); } } class MyCallable implements Callable<Object> { private String taskNum; MyCallable(String taskNum) { this.taskNum = taskNum; } public Object call() throws Exception { System.out.println(">>>" + taskNum + "任务启动"); Date dateTmp1 = new Date(); Thread.sleep(1000); Date dateTmp2 = new Date(); long time = dateTmp2.getTime() - dateTmp1.getTime(); System.out.println(">>>" + taskNum + "任务终止"); return taskNum + "任务返回运行结果,当前任务时间【" + time + "毫秒】"; } }
The main difference is that the run method of the Runnable interface has no return value;
The call method of the Callable interface has a return value and supports the generic Runnable interface. The run method can only throw runtime exceptions and cannot be captured and processed;
The call method of the Callable interface allows exceptions to be thrown and exception information can be obtained
The thread object calls the run method without starting the thread. Only object calls methods.
The thread object calls start to open the thread, and lets the jvm call the run method to execute in the opened thread. Calling the start method can start the thread and make the thread enter the ready state, and the run method is just a normal method of thread, or Executed in the main thread.
The basic methods related to threads include wait, notify, notifyAll, sleep, join, yield, etc.
Thread wait (wait) calls this The thread of the method enters the waiting state and will only return if it waits for notification from another thread or is interrupted. It should be noted that after calling the wait() method, the object's lock will be released. Therefore, the wait method is generally used in synchronized methods or synchronized code blocks.
Thread sleep (sleep) sleep causes the current thread to sleep. Unlike the wait method, sleep will not release the currently occupied lock. sleep(long) will cause the thread to enter the TIMED-WATING state, and the wait() method Will cause the current thread to enter the WATING state.
Thread yield (yield) yield will cause the current thread to yield the CPU execution time slice and re-compete with other threads for the CPU time slice. Generally speaking, threads with high priority have a greater chance of successfully competing for CPU time slices, but this is not absolute. Some operating systems are not sensitive to thread priority.
Thread interrupt (interrupt) Interrupting a thread is intended to give the thread a notification signal, which will affect an interrupt flag inside the thread. This thread itself will not change the state (such as blocking, termination, etc.) because of this
Join waits for other threads to terminate the join() method, waits for other threads to terminate, and calls the join() method of a thread in the current thread. , then the current thread changes to the blocking state and returns to another thread to end. The current thread changes from the blocking state to the ready state again, waiting for the favor of the CPU.
Thread wake-up (notify) The notify() method in the Object class wakes up a single thread waiting on this object monitor. If all threads are waiting on this object, one of the threads will be chosen to wake up. The choice is arbitrary. and occurs when an implementation decision is made, the thread waits on the object's monitor by calling one of the wait() methods until the current thread relinquishes the lock on this object before it can continue executing the awakened thread. The awakened thread will compete in the normal manner with all other threads actively synchronizing on the object. Another similar method is notifyAll(), which wakes up all threads waiting on the same monitor.
① From different classes wait(): from Object class; sleep(): from Thread class;
② Regarding the release of locks: wait(): the lock will be released during the waiting process; sleep(): the lock will not be released during the waiting process
③ Scope of use: wait(): must be synchronized Used in code blocks; sleep(): can be used anywhere;
④ Whether to catch exceptions wait(): No need to catch exceptions; sleep(): Need to catch exceptions;
Multi-threading principle: Multi-threading is performed in a concurrent manner. For a CPU, it can only execute one program at a certain point in time, that is, it can only run one process at the same time. The CPU will continuously switch between these processes, and each thread will execute for one time. Because the execution speed of the CPU is too fast relative to our perception, although the CPU rotates execution between multiple processes, we feel as if multiple processes are executing at the same time.
The CPU will switch between multiple processes. If we open too many programs, the time it takes for the CPU to switch to each process will also become longer, and we will also feel that the machine is running slower. Although reasonable use of multi-threading can improve efficiency, excessive use does not bring about efficiency improvements.
Multi-threading technology mainly solves the problem of multiple thread execution in the processor unit. It can significantly reduce the idle time of the processor unit and increase the throughput capacity of the processor unit.
The above is the detailed content of What are the ways to implement multithreading in Java?. For more information, please follow other related articles on the PHP Chinese website!