溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java線程的創(chuàng)建方式有哪些

發(fā)布時(shí)間:2021-11-29 09:20:40 來源:億速云 閱讀:148 作者:iii 欄目:開發(fā)技術(shù)

這篇文章主要講解了“Java線程的創(chuàng)建方式有哪些”,文中的講解內(nèi)容簡單清晰,易于學(xué)習(xí)與理解,下面請(qǐng)大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“Java線程的創(chuàng)建方式有哪些”吧!

1、Thread

繼承Thread類,并重寫run方法

class ThreadDemo1 extends Thread {

  @Override

  public void run() {

    log.info("{}", Thread.currentThread().getName());

  }

}

線程啟動(dòng)方式:

ThreadDemo1 t1 = new ThreadDemo1();

t1.setName("t1");

t1.start();

簡便寫法:

Thread t1 = new Thread() {

    @Override

    public void run() {

        log.info("{}", Thread.currentThread().getName());

    }

};

t1.setName("t1");

t1.start();

2、Runnable和Thread

Thread類的構(gòu)造函數(shù)支持傳入Runnable的實(shí)現(xiàn)類

public Thread(Runnable target) {

    init(null, target, "Thread-" + nextThreadNum(), 0);

}

Thread(Runnable target, AccessControlContext acc) {

    init(null, target, "Thread-" + nextThreadNum(), 0, acc, false);

}

Runnable是一個(gè)函數(shù)式接口(FunctionalInterface

@FunctionalInterface

public interface Runnable {

    // 沒有返回值

    public abstract void run();

}

因此需要?jiǎng)?chuàng)建類實(shí)現(xiàn)Runnable接口,重寫run方法

class ThreadDemo2 implements Runnable {

    @Override

    public void run() {

        log.info("{}", Thread.currentThread().getName());

    }

}

簡便寫法:

Thread t2 = new Thread(() -> log.info("{}", Thread.currentThread().getName()), "t2");

t2.start();

3、Runnable和Thread

CallableRunnable一樣,也是一個(gè)函數(shù)式接口,二者的區(qū)別非常明顯,Runnablerun方法沒有返回值,Callable中的run方法有返回值(可以通過泛型約束返回值類型)。因此在需要獲取線程執(zhí)行的返回值時(shí),可以使用Callable

@FunctionalInterface

public interface Callable<V> {

    // 帶返回值

    V call() throws Exception;

}

Thread的構(gòu)造函數(shù)中,并沒有看到Callable,只有Runnable

Java線程的創(chuàng)建方式有哪些

此時(shí)需要一個(gè)可以提交Callable給Thread的類,這類就是FutureTask;FutureTask實(shí)現(xiàn)類Runnable接口。

Java線程的創(chuàng)建方式有哪些

并且FutureTask提供了傳入Callable的構(gòu)造函數(shù)

public FutureTask(Callable<V> callable) {

    if (callable == null)

        throw new NullPointerException();

    this.callable = callable;

    this.state = NEW;       // ensure visibility of callable

}

因此可以通過FutureTask傳入Callable實(shí)現(xiàn),再將FutureTask傳給Thread即可

ThreadDemo3 implements Callable<Integer> {

    @Override

    public Integer call() throws Exception {

        log.info("{}", Thread.currentThread().getName());

        return 1998;

    }

}
// Callable 實(shí)現(xiàn)類

ThreadDemo3 callable = new ThreadDemo3();

// 通過Callable創(chuàng)建FutureTask

FutureTask<Integer> task = new FutureTask(callable);

// 通過FutureTask創(chuàng)建Thread

Thread t3 = new Thread(task, "t3");

t3.start();

簡便寫法:

Thread t3 = new Thread(new FutureTask<Integer>(() -> {

    log.info("{}", Thread.currentThread().getName());

    return 1998;

}), "t3");

t3.start();

4、三者對(duì)比

創(chuàng)建線程的方式有三種:

Thread、Runnable+ThreadCallable+FutureTask+Thread;這三者如何選擇呢?

  • 首先在實(shí)際的開發(fā)過程中,我們不會(huì)直接創(chuàng)建線程,因?yàn)轭l繁創(chuàng)建和銷毀線程開銷比較大,此外不利于管理和釋放,因此項(xiàng)目中都是通過設(shè)計(jì)線程池來管理線程資源

  • Thread、Runnable+Thread相比,Runnable+Thread將線程的創(chuàng)建和任務(wù)模塊解耦了,代碼設(shè)計(jì)更加靈活,此外更加利于任務(wù)的提交,更方便和線程池結(jié)合使用

  • Callable+FutureTask+Thread適用于需要獲取線程返回結(jié)果的場景

5、注意項(xiàng)

文中多次使用thread.start();需要注意的是,調(diào)用線程的start()方法表示啟動(dòng)線程,但是線程是否執(zhí)行并不確定,這需要操作系統(tǒng)調(diào)度,線程分配到CPU執(zhí)行時(shí)間片才能執(zhí)行。多核CPU下多個(gè)線程同時(shí)啟動(dòng),線程之間交替執(zhí)行,執(zhí)行順序是不確定的。

感謝各位的閱讀,以上就是“Java線程的創(chuàng)建方式有哪些”的內(nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對(duì)Java線程的創(chuàng)建方式有哪些這一問題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是億速云,小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI