亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java創(chuàng)建線程及配合使用Lambda方式

 更新時(shí)間:2021年08月24日 11:59:18   作者:FXBStudy  
這篇文章主要介紹了Java創(chuàng)建線程及配合使用Lambda方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

一、創(chuàng)建線程三種方式

1.1 繼承Thread類創(chuàng)建線程類

  • 定義Thread類的子類,并重寫該類的run方法,該run方法的方法體就代表了線程要完成的任務(wù)。因此把run()方法稱為執(zhí)行體。
  • 創(chuàng)建Thread子類的實(shí)例,即創(chuàng)建了線程對(duì)象。
  • 調(diào)用線程對(duì)象的start()方法來啟動(dòng)該線程。
public class FirstThreadTest extends Thread {
    int i = 0;
    // 重寫run方法,run方法的方法體就是現(xiàn)場(chǎng)執(zhí)行體
    public void run() {
        for (; i < 5; i++) {
            System.out.println(getName() + "  " + i);
        }
    }
    public static void main(String[] args) {
        for (int i = 0; i < 3; i++) {
            System.out.println(Thread.currentThread().getName() + "  : " + i);
            if (i == 2) {
                new FirstThreadTest().start();
                new FirstThreadTest().start();
            }
        }
    }
}

上述代碼中Thread.currentThread()方法返回當(dāng)前正在執(zhí)行的線程對(duì)象。GetName()方法返回調(diào)用該方法的線程的名字。

1.2 通過Runnable接口創(chuàng)建線程類

  • 定義runnable接口的實(shí)現(xiàn)類,并重寫該接口的run()方法,該run()方法的方法體同樣是該線程的線程執(zhí)行體。
  • 創(chuàng)建 Runnable實(shí)現(xiàn)類的實(shí)例,并以此實(shí)例作為Thread的target來創(chuàng)建Thread對(duì)象,該Thread對(duì)象才是真正的線程對(duì)象。
  • 調(diào)用線程對(duì)象的start()方法來啟動(dòng)該線程。
public class RunnableThreadTest implements Runnable {
    private int i;
    public void run() {
        for (i = 0; i < 5; i++) {
            System.out.println(Thread.currentThread().getName() + " " + i);
        }
    }
    public static void main(String[] args) {
        for (int i = 0; i < 3; i++) {
            System.out.println(Thread.currentThread().getName() + " " + i);
            if (i == 2) {
                RunnableThreadTest rtt = new RunnableThreadTest();
                new Thread(rtt, "新線程1").start();
                new Thread(rtt, "新線程2").start();
            }
        }
    }
}

線程的執(zhí)行流程很簡(jiǎn)單,當(dāng)執(zhí)行代碼start()時(shí),就會(huì)執(zhí)行對(duì)象中重寫的void run();方法,該方法執(zhí)行完成后,線程就消亡了。

使用Lambda表達(dá)式

public class RunnableThreadTest {
    // 目的是為了代碼的重用【靜態(tài)方法】
    public static void threadRunCode_Static() {
        for (int i = 0; i < 5; i++) {
            System.out.println(Thread.currentThread().getName() + " " + i);
        }
    }
    // 目的是為了代碼的重用【非靜態(tài)方法】
    public void threadRunCode() {
        for (int i = 0; i < 5; i++) {
            System.out.println(Thread.currentThread().getName() + " " + i);
        }
    }
    @Test
    public void testStatic() {
        // 重用靜態(tài)方法中的代碼【使用方法引用】
        for (int i = 0; i < 3; i++) {
            System.out.println(Thread.currentThread().getName() + " " + i);
            if (i == 2) {
                new Thread(RunnableThreadTest::threadRunCode_Static, "線程1").start();
                ;
                new Thread(RunnableThreadTest::threadRunCode_Static, "線程2").start();
                ;
            }
        }
    }
    @Test
    public void testNoStatic() {
        // 重用非靜態(tài)方法中的代碼【使用方法引用】
        RunnableThreadTest temp = new RunnableThreadTest();
        for (int i = 0; i < 3; i++) {
            System.out.println(Thread.currentThread().getName() + " " + i);
            if (i == 2) {
                new Thread(temp::threadRunCode, "線程1").start();
                new Thread(temp::threadRunCode, "線程2").start();
            }
        }
    }
    @Test
    public void testLambda() {
        // 重用靜態(tài)方法中的代碼【使用方法引用】
        for (int i = 0; i < 3; i++) {
            System.out.println(Thread.currentThread().getName() + " " + i);
            if (i == 2) {
                new Thread(() -> {
                    for (int b = 0; b < 5; b++) {
                        System.out.println(Thread.currentThread().getName() + " " + b);
                    }
                },"線程1").start();
                new Thread(() -> {
                    for (int b = 0; b < 5; b++) {
                        System.out.println(Thread.currentThread().getName() + " " + b);
                    }
                },"線程2").start();
            }
        }
    }
}

1.3 通過Callable和Future創(chuàng)建線程

public interface Callable{
  V call() throws Exception;
}
  • 創(chuàng)建Callable接口的實(shí)現(xiàn)類,并實(shí)現(xiàn)call()方法,該call()方法將作為線程執(zhí)行體,并且有返回值。
  • 創(chuàng)建Callable實(shí)現(xiàn)類的實(shí)例,使用FutureTask類來包裝Callable對(duì)象,該FutureTask對(duì)象封裝了該Callable對(duì)象的call()方法的返回值。(FutureTask是一個(gè)包裝器,它通過接受Callable來創(chuàng)建,它同時(shí)實(shí)現(xiàn)了Future和Runnable接口。)
  • 使用FutureTask對(duì)象作為Thread對(duì)象的target創(chuàng)建并啟動(dòng)新線程。
  • 調(diào)用FutureTask對(duì)象的get()方法來獲得子線程執(zhí)行結(jié)束后的返回值
public class CallableThreadTest implements Callable<Integer> {
    @Override
    public Integer call() throws Exception {
        int i = 0;
        for (; i < 5; i++) {
            System.out.println(Thread.currentThread().getName() + " " + i);
        }
        return i;
    }
    public static void main(String[] args) {
        CallableThreadTest ctt = new CallableThreadTest();
        FutureTask<Integer> ft = new FutureTask<>(ctt);
        for (int i = 0; i < 3; i++) {
            System.out.println(Thread.currentThread().getName() + " 的循環(huán)變量i的值" + i);
            if (i == 2) {
                new Thread(ft, "有返回值的線程").start();
            }
        }
        try {
            System.out.println("子線程的返回值:" + ft.get());
        } catch (InterruptedException | ExecutionException e) {
            e.printStackTrace();
        }
    }
}

使用Lambda表達(dá)式

public class CallableThreadTest {
    public static void main(String[] args) {
        FutureTask<Integer> ft = new FutureTask<>(() -> {
            int i = 0;
            for (; i < 5; i++) {
                System.out.println(Thread.currentThread().getName() + " " + i);
            }
            return i;
        });
        for (int i = 0; i < 3; i++) {
            System.out.println(Thread.currentThread().getName() + " 的循環(huán)變量i的值" + i);
            if (i == 2) {
                new Thread(ft, "有返回值的線程").start();
            }
        }
        try {
            System.out.println("子線程的返回值:" + ft.get());
        } catch (InterruptedException | ExecutionException e) {
            e.printStackTrace();
        }
    }
}

二、創(chuàng)建線程的三種方式的對(duì)比

2.1 實(shí)現(xiàn)Runnable、Callable接口的方式創(chuàng)建多線程

優(yōu)勢(shì):

  • 線程類只是實(shí)現(xiàn)了Runnable接口或Callable接口,還可以繼承其他類。
  • 在這種方式下,多個(gè)線程可以共享同一個(gè)target對(duì)象,所以非常適合多個(gè)相同線程來處理同一份資源的情況,從而可以將CPU、代碼和數(shù)據(jù)分開,形成清晰的模型,較好地體現(xiàn)了面向?qū)ο蟮乃枷搿?/li>

劣勢(shì):

  • 編程稍微復(fù)雜,如果要訪問當(dāng)前線程,則必須使用Thread.currentThread()方法。

2.2 繼承Thread類的方式創(chuàng)建多線程

優(yōu)勢(shì):

  • 編寫簡(jiǎn)單,如果需要訪問當(dāng)前線程,則無需使用Thread.currentThread()方法,直接使用this即可獲得當(dāng)前線程。

劣勢(shì):

  • 線程類已經(jīng)繼承了Thread類,所以不能再繼承其他父類。

2.3 Runnable和Callable的區(qū)別

  • Callable規(guī)定(重寫)的方法是call(),Runnable規(guī)定(重寫)的方法是run()。
  • Callable的任務(wù)執(zhí)行后可返回值,而Runnable的任務(wù)是不能返回值的。
  • call方法可以拋出異常,run方法不可以。
  • 運(yùn)行Callable任務(wù)可以拿到一個(gè)Future對(duì)象,表示異步計(jì)算的結(jié)果。它提供了檢查計(jì)算是否完成的方法,以等待計(jì)算的完成,并檢索計(jì)算的結(jié)果。通過Future對(duì)象可以了解任務(wù)執(zhí)行情況,可取消任務(wù)的執(zhí)行,還可獲取執(zhí)行結(jié)果。

拓展:

Lambda表達(dá)式的強(qiáng)大之處就是傳遞代碼,而Runnable和Callable接口都是符合Lambda要求的函數(shù)式接口。因此,我們可以不用創(chuàng)建這兩個(gè)接口的實(shí)現(xiàn)類,而是直接將其中的實(shí)現(xiàn)代碼傳遞到Thread的target即可。

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • 使用Java注解模擬spring ioc容器過程解析

    使用Java注解模擬spring ioc容器過程解析

    這篇文章主要介紹了使用Java注解模擬spring ioc容器過程解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-07-07
  • 詳細(xì)聊聊Mybatis中萬能的Map

    詳細(xì)聊聊Mybatis中萬能的Map

    最近有個(gè)需求,就是使用mybatis時(shí),向mysql中插入數(shù)據(jù),其參數(shù)為map類型,下面這篇文章主要給大家介紹了關(guān)于Mybatis中萬能的Map的相關(guān)資料,需要的朋友可以參考下
    2021-12-12
  • Java util concurrent及基本線程原理簡(jiǎn)介

    Java util concurrent及基本線程原理簡(jiǎn)介

    這篇文章主要介紹了Java util concurrent及基本線程原理簡(jiǎn)介,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-04-04
  • Spring Boot如何動(dòng)態(tài)創(chuàng)建Bean示例代碼

    Spring Boot如何動(dòng)態(tài)創(chuàng)建Bean示例代碼

    這篇文章主要給大家介紹了關(guān)于Spring Boot如何動(dòng)態(tài)創(chuàng)建Bean的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。
    2017-09-09
  • Java如何實(shí)現(xiàn)長連接

    Java如何實(shí)現(xiàn)長連接

    這篇文章主要介紹了Java如何實(shí)現(xiàn)長連接,幫助大家更好的理解和學(xué)習(xí)Java,感興趣的朋友可以了解下
    2020-09-09
  • Java算法之堆排序代碼示例

    Java算法之堆排序代碼示例

    這篇文章主要介紹了Java算法之堆排序代碼示例,具有一定參考價(jià)值,需要的朋友可以了解下。
    2017-11-11
  • Java中綴表達(dá)式轉(zhuǎn)后綴表達(dá)式流程詳解

    Java中綴表達(dá)式轉(zhuǎn)后綴表達(dá)式流程詳解

    中綴表達(dá)式是一個(gè)通用的算術(shù)或邏輯公式表示方法。,中綴表達(dá)式不容易被計(jì)算機(jī)解析,但仍被許多程序語言使用,因?yàn)樗先藗兊钠毡橛梅?。本文介紹了實(shí)現(xiàn)中綴表達(dá)式的方法,需要的可以參考一下
    2022-09-09
  • Java Socket報(bào)錯(cuò)打開文件過多的問題

    Java Socket報(bào)錯(cuò)打開文件過多的問題

    這篇文章主要介紹了Java Socket報(bào)錯(cuò)打開文件過多的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-05-05
  • java發(fā)起http請(qǐng)求調(diào)用post與get接口的方法實(shí)例

    java發(fā)起http請(qǐng)求調(diào)用post與get接口的方法實(shí)例

    在實(shí)際開發(fā)過程中,我們經(jīng)常需要調(diào)用對(duì)方提供的接口或測(cè)試自己寫的接口是否合適,下面這篇文章主要給大家介紹了關(guān)于java發(fā)起http請(qǐng)求調(diào)用post與get接口的相關(guān)資料,需要的朋友可以參考下
    2022-08-08
  • SpringBoot實(shí)現(xiàn)快遞物流查詢功能(快遞鳥)

    SpringBoot實(shí)現(xiàn)快遞物流查詢功能(快遞鳥)

    本文將基于springboot2.4.0實(shí)現(xiàn)快遞物流查詢,物流信息的獲取通過快遞鳥第三方實(shí)現(xiàn),本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧
    2021-10-10

最新評(píng)論