一、繼承Thread類創(chuàng)建
通過繼承Thread并且重寫其run(),run方法中即線程執(zhí)行任務(wù)。創(chuàng)建后的子類通過調(diào)用 start() 方法即可執(zhí)行線程方法。
通過繼承Thread實(shí)現(xiàn)的線程類,多個線程間無法共享線程類的實(shí)例變量。(需要創(chuàng)建不同Thread對象,自然不共享)
例子:
/**
* 通過繼承Thread實(shí)現(xiàn)線程
*/
public class ThreadTest extends Thread{
private int i = 0 ;
@Override
public void run() {
for(;i<50;i++){
System.out.println(Thread.currentThread().getName() + " is running " + i );
}
}
public static void main(String[] args) {
for(int j=0;j<50;j++){if(j=20){
new ThreadTest().start() ;
new ThreadTest().start() ;
}
}
}
}
二、 通過Runnable接口創(chuàng)建線程類
該方法需要先 定義一個類實(shí)現(xiàn)Runnable接口,并重寫該接口的 run() 方法,此run方法是線程執(zhí)行體。接著創(chuàng)建 Runnable實(shí)現(xiàn)類的對象,作為創(chuàng)建Thread對象的參數(shù)target,此Thread對象才是真正的線程對象。通過實(shí)現(xiàn)Runnable接口的線程類,是互相共享資源的。
/**
* 通過實(shí)現(xiàn)Runnable接口實(shí)現(xiàn)的線程類
*/
public class RunnableTest implements Runnable {
private int i ;
@Override
public void run() {
for(;i<50;i++){
System.out.println(Thread.currentThread().getName() + " -- " + i);
}
}
public static void main(String[] args) {
for(int i=0;i<100;i++){
System.out.println(Thread.currentThread().getName() + " -- " + i);
if(i==20){
RunnableTest runnableTest = new RunnableTest() ;
new Thread(runnableTest,"線程1").start() ;
new Thread(runnableTest,"線程2").start() ;
}
}
}
}
三、 使用Callable和Future創(chuàng)建線程
從繼承Thread類和實(shí)現(xiàn)Runnable接口可以看出,上述兩種方法都不能有返回值,且不能聲明拋出異常。而Callable接口則實(shí)現(xiàn)了此兩點(diǎn),Callable接口如同Runable接口的升級版,其提供的call()方法將作為線程的執(zhí)行體,同時允許有返回值。
但是Callable對象不能直接作為Thread對象的target,因為Callable接口是 JAVA 5 新增的接口,不是Runnable接口的子接口。對于這個問題的解決方案,就引入 Future接口,此接口可以接受call() 的返回值,RunnableFuture接口是Future接口和Runnable接口的子接口,可以作為Thread對象的target 。并且, Future 接口提供了一個實(shí)現(xiàn)類:FutureTask 。
FutureTask實(shí)現(xiàn)了RunnableFuture接口,可以作為 Thread對象的target。
例子:
import java.util.concurrent.Callable;
import java.util.concurrent.FutureTask;
public class CallableTest {
public static void main(String[] args) {
CallableTest callableTest = new CallableTest() ;
//因為Callable接口是函數(shù)式接口,可以使用Lambda表達(dá)式
FutureTask<Integer> task = new FutureTask<Integer>((Callable<Integer>)()->{
int i = 0 ;
for(;i<100;i++){
System.out.println(Thread.currentThread().getName() + "的循環(huán)變量i的值 :" + i);
}
return i;
});
for(int i=0;i<100;i++){
System.out.println(Thread.currentThread().getName()+" 的循環(huán)變量i : + i");
if(i==20){
new Thread(task,"有返回值的線程").start();
}
}
try{
System.out.println("子線程返回值 : " + task.get());
}catch (Exception e){
e.printStackTrace();
}
}
}
總結(jié)
通過上述三種方式,其實(shí)可以歸為兩類:繼承類和實(shí)現(xiàn)接口兩種方式。相比繼承, 接口實(shí)現(xiàn)可以更加靈活,不會受限于Java的單繼承機(jī)制。并且通過實(shí)現(xiàn)接口的方式可以共享資源,適合多線程處理同一資源的情況。線程知識豐富繁雜,更多細(xì)節(jié)還需努力學(xué)習(xí)掌握。






