免费视频淫片aa毛片_日韩高清在线亚洲专区vr_日韩大片免费观看视频播放_亚洲欧美国产精品完整版

打開APP
userphoto
未登錄

開通VIP,暢享免費電子書等14項超值服

開通VIP
【第77題】JAVA高級技術(shù)-多線程11(創(chuàng)建線程的5種方式)

回城傳送–》《JAVA筑基100例》

文章目錄

零、前言

? 今天是學習 JAVA語言 打卡的第77天,每天我會提供一篇文章供群成員閱讀( 不需要訂閱付錢 ),讀完文章之后,按解題思路,自己再實現(xiàn)一遍。在小虛竹JAVA社區(qū) 中對應的 【打卡貼】打卡,今天的任務就算完成了。

? 因為大家都在一起學習同一篇文章,所以有什么問題都可以在群里問,群里的小伙伴可以迅速地幫到你,一個人可以走得很快,一群人可以走得很遠,有一起學習交流的戰(zhàn)友,是多么幸運的事情。

? 學完后,自己寫篇學習報告的博客,可以發(fā)布到小虛竹JAVA社區(qū) ,供學弟學妹們參考。

? 我的學習策略很簡單,題海策略+ 費曼學習法。如果能把這100題都認認真真自己實現(xiàn)一遍,那意味著 JAVA語言 已經(jīng)筑基成功了。后面的進階學習,可以繼續(xù)跟著我,一起走向架構(gòu)師之路。

一、題目描述

題目:

Java創(chuàng)建線程的幾種方式:

Java使用Thread類代表線程,所有線程對象都必須是Thread類或者其子類的實例。Java可以用以下5種方式來創(chuàng)建線程:

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

2)實現(xiàn)Runnable接口創(chuàng)建線程;

3)實現(xiàn)Callable接口,通過FutureTask包裝器來創(chuàng)建Thread線程;

4)使用ExecutorService、Callable(或者Runnable)、Future實現(xiàn)由返回結(jié)果的線程。

5)使用CompletableFuture類創(chuàng)建異步線程,且是據(jù)有返回結(jié)果的線程。 JDK8新支持的

實現(xiàn):使用這5種方式創(chuàng)建線程,體驗其中的妙處。

二、解題思路

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

Thread類本質(zhì)上是實現(xiàn)了Runnable接口的一個實例,代表一個線程的實例。啟動線程的唯一方法就是通過Thread類的start()實例方法。start()方法是一個native方法,它將啟動一個新線程,并執(zhí)行run()方法。這種方式實現(xiàn)多線程很簡單,通過自己的類直接extends Thread,并復寫run()方法,就可以啟動新線程并執(zhí)行自己定義的run()方法。

實現(xiàn)Runnable接口創(chuàng)建線程

如果自己的類已經(jīng)extends另一個類,就無法直接extends Thread,此時,可以實現(xiàn)一個Runnable接口

實現(xiàn)Callable接口,通過FutureTask包裝器來創(chuàng)建Thread線程

實現(xiàn)一個Callable接口(它是一個具有返回值的)

使用ExecutorService、Callable(或者Runnable)、Future實現(xiàn)由返回結(jié)果的線程

Executors類,提供了一系列工廠方法用于創(chuàng)建線程池,返回的線程池都實現(xiàn)了ExecutorService接口:

//創(chuàng)建固定數(shù)目線程的線程池。
public static ExecutorService newFixedThreadPool(int nThreads) ;
//創(chuàng)建一個可緩存的線程池,調(diào)用execute 將重用以前構(gòu)造的線程(如果線程可用)。如果現(xiàn)有線程沒有可用的,則創(chuàng)建一個新線程并添加到池中。終止并從緩存中移除那些已有 60 秒鐘未被使用的線程。
public static ExecutorService newCachedThreadPool();
//創(chuàng)建一個單線程化的Executor。
public static ExecutorService newSingleThreadExecutor();
//創(chuàng)建一個支持定時及周期性的任務執(zhí)行的線程池,多數(shù)情況下可用來替代Timer類。
public static ScheduledExecutorService newScheduledThreadPool(int corePoolSize);

ExecutoreService提供了submit()方法,傳遞一個Callable,或Runnable,返回Future。如果Executor后臺線程池還沒有完成Callable的計算,這調(diào)用返回Future對象的get()方法,會阻塞直到計算完成。

使用CompletableFuture類創(chuàng)建異步線程,且是據(jù)有返回結(jié)果的線程

Future模式的缺點

Future雖然可以實現(xiàn)獲取異步執(zhí)行結(jié)果的需求,但是它沒有提供通知的機制,我們無法得知Future什么時候完成。

要么使用阻塞,在future.get()的地方等待future返回的結(jié)果,這時又變成同步操作。要么使用isDone()輪詢地判斷Future是否完成,這樣會耗費CPU的資源。

CompletableFuture介紹

JDK1.8新加入的一個實現(xiàn)類CompletableFuture,實現(xiàn)了Future, CompletionStage兩個接口。

CompletableFuture中4個異步執(zhí)行任務靜態(tài)方法:

public static <U> CompletableFuture<U> supplyAsync(Supplier<U> supplier) {
        return asyncSupplyStage(asyncPool, supplier);
    }
    
public static <U> CompletableFuture<U> supplyAsync(Supplier<U> supplier,Executor executor) {
    return asyncSupplyStage(screenExecutor(executor), supplier);
}
    
public static CompletableFuture<Void> runAsync(Runnable runnable) {
    return asyncRunStage(asyncPool, runnable);
}
    
public static CompletableFuture<Void> runAsync(Runnable runnable, Executor executor) {
    return asyncRunStage(screenExecutor(executor), runnable);
}

其中supplyAsync用于有返回值的任務,runAsync則用于沒有返回值的任務。Executor參數(shù)可以手動指定線程池,否則默認ForkJoinPool.commonPool()系統(tǒng)級公共線程池

三、代碼詳解

第一種:繼承Thread類創(chuàng)建線程

package com.xiaoxuzhu;

import java.util.concurrent.CountDownLatch;

/**
 * Description:繼承Thread類創(chuàng)建線程
 *
 * @author xiaoxuzhu
 * @version 1.0
 *
 * <pre>
 * 修改記錄:
 * 修改后版本        修改人修改日期修改內(nèi)容
 * 2022/5/15.1    xiaoxuzhu2022/5/15    Create
 * </pre>
 * @date 2022/5/15
 */
public class ThreadDemo1 extends Thread {
    CountDownLatch countDownLatch;

    public ThreadDemo1(CountDownLatch countDownLatch) {
        this.countDownLatch = countDownLatch;
    }

    @Override
    public void run() {
        try {
            Thread.sleep(2000);
            System.out.println(Thread.currentThread().getName() + ":my thread ");
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            countDownLatch.countDown();
        }
    }

    public static void main(String[] args) {
        // 第一種:使用extends Thread方式
        CountDownLatch countDownLatch1 = new CountDownLatch(2);
        for (int i = 0; i < 2; i++) {
            ThreadDemo1 myThread1 = new ThreadDemo1(countDownLatch1);
            myThread1.start();
        }

        try {
            countDownLatch1.await();
            System.out.println("thread complete...");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }


}


第二種:實現(xiàn)Runnable接口創(chuàng)建線程

package com.xiaoxuzhu;

import java.util.concurrent.CountDownLatch;

/**
 * Description: 實現(xiàn)Runnable接口創(chuàng)建線程
 *
 * @author xiaoxuzhu
 * @version 1.0
 *
 * <pre>
 * 修改記錄:
 * 修改后版本        修改人修改日期修改內(nèi)容
 * 2022/5/15.1    xiaoxuzhu2022/5/15    Create
 * </pre>
 * @date 2022/5/15
 */
public class ThreadDemo2  implements Runnable{
    CountDownLatch countDownLatch;

    public ThreadDemo2(CountDownLatch countDownLatch) {
        this.countDownLatch = countDownLatch;
    }
    @Override
    public void run() {
        try {
            Thread.sleep(2000);
            System.out.println(Thread.currentThread().getName() + ":my runnable ");
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            countDownLatch.countDown();
        }
    }

    public static void main(String[] args) {
        // 第二種:使用implements Runnable方式
        CountDownLatch countDownLatch2 = new CountDownLatch(2);
        ThreadDemo2 myRunnable = new ThreadDemo2(countDownLatch2);
        for (int i = 0; i < 2; i++) {
            new Thread(myRunnable).start();
        }

        try {
            countDownLatch2.await();
            System.out.println("runnable complete...");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}


第三種:實現(xiàn)Callable接口,通過FutureTask包裝器來創(chuàng)建Thread線程

計算1~100的疊加

package com.xiaoxuzhu;

import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;

/**
 * Description: 實現(xiàn)Callable接口,通過FutureTask包裝器來創(chuàng)建Thread線程
 *  跟Runnable比,不同點在于它是一個具有返回值的,且會拋出異常
 *  //用futureTask接收結(jié)果
 *
 * @author xiaoxuzhu
 * @version 1.0
 *
 * <pre>
 * 修改記錄:
 * 修改后版本        修改人修改日期修改內(nèi)容
 * 2022/5/15.1    xiaoxuzhu2022/5/15    Create
 * </pre>
 * @date 2022/5/15
 */
public class ThreadDemo3 implements Callable<Integer> {

    public static void main(String[] args) {
        ThreadDemo3 threadDemo03 = new ThreadDemo3();
        //1、用futureTask接收結(jié)果
        FutureTask<Integer> futureTask = new FutureTask<>(threadDemo03);
        new Thread(futureTask).start();

        //2、接收線程運算后的結(jié)果
        try {
            //futureTask.get();這個是堵塞性的等待
            Integer sum = futureTask.get();
            System.out.println("sum="+sum);
            System.out.println("-------------------");
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (ExecutionException e) {
            e.printStackTrace();
        }
    }

    @Override
    public Integer call() throws Exception {
        int sum = 0;
        for (int i = 0; i <101 ; i++) {
            sum+=i;
        }
        return sum;
    }
}


第四種:使用ExecutorService、Callable(或者Runnable)、Future實現(xiàn)由返回結(jié)果的線程

package com.xiaoxuzhu;

import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.concurrent.Callable;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;

/**
 * Description: 使用ExecutorService、Callable(或者Runnable)、Future實現(xiàn)由返回結(jié)果的線程
 *
 * @author xiaoxuzhu
 * @version 1.0
 *
 * <pre>
 * 修改記錄:
 * 修改后版本        修改人修改日期修改內(nèi)容
 * 2022/5/15.1    xiaoxuzhu2022/5/15    Create
 * </pre>
 * @date 2022/5/15
 */
public class ThreadDemo4 {

    static class MyCallable implements Callable<Integer> {
        private CountDownLatch countDownLatch;

        public MyCallable(CountDownLatch countDownLatch) {
            this.countDownLatch = countDownLatch;
        }

        public Integer call() {
            int sum = 0;
            try {
               
                for (int i = 0; i <= 100; i++) {
                    sum += i;
                }
                System.out.println("線程執(zhí)行結(jié)果:"+sum);
                
            } finally {
                countDownLatch.countDown();
            }
            return sum;
        }

    }

    public static void main(String[] args) throws ExecutionException, InterruptedException {
        // 第四種:使用使用線程池方式
        // 接受返回參數(shù)
        List<Future> resultItems2 = new ArrayList<Future>();
        // 給線程池初始化5個線程
        ExecutorService executorService = Executors.newFixedThreadPool(5);
        CountDownLatch countDownLatch4 = new CountDownLatch(10);

        for (int i = 0; i < 10; i++) {
            MyCallable myCallable = new MyCallable(countDownLatch4);
            Future result = executorService.submit(myCallable);
            resultItems2.add(result);
        }

        // 等待線程池中分配的任務完成后才關(guān)閉(關(guān)閉之后不允許有新的線程加入,但是它并不會等待線程結(jié)束),
        // 而executorService.shutdownNow();是立即關(guān)閉不管是否線程池中是否有其他未完成的線程。
        executorService.shutdown();
        try {
            countDownLatch4.await();
            Iterator<Future> iterator = resultItems2.iterator();
            System.out.println("----------------------");
            while (iterator.hasNext()) {
                try {
                    System.out.println("線程返回結(jié)果:"+iterator.next().get());
                } catch (ExecutionException e) {
                    e.printStackTrace();
                }
            }
            System.out.println("callable complete...");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}


第五種:使用CompletableFuture類創(chuàng)建異步線程,且是據(jù)有返回結(jié)果的線程

package com.xiaoxuzhu;

import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;

import org.junit.Test;

/**
 * Description:  使用CompletableFuture類創(chuàng)建異步線程,且是據(jù)有返回結(jié)果的線程。
 *
 * @author xiaoxuzhu
 * @version 1.0
 *
 * <pre>
 * 修改記錄:
 * 修改后版本        修改人修改日期修改內(nèi)容
 * 2022/5/15.1    xiaoxuzhu2022/5/15    Create
 * </pre>
 * @date 2022/5/15
 */
public class ThreadDemo5 {

    /**
     * A任務B任務完成后,才執(zhí)行C任務
     * 返回值的處理
     * @param
     *@return void
     **/
    @Test
    public void completableFuture1(){
        CompletableFuture<String> future1 = CompletableFuture.supplyAsync(() -> {
            try {
                Thread.sleep(10);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("future1 finished!");
            return "future1 finished!";
        });

        CompletableFuture<String> future2 = CompletableFuture.supplyAsync(() -> {
            System.out.println("future2 finished!");
            return "future2 finished!";
        });

        CompletableFuture<Void> future3 = CompletableFuture.allOf(future1, future2);
        try {
            future3.get();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (ExecutionException e) {
            e.printStackTrace();
        }
        System.out.println("future1: " + future1.isDone() + " future2: " + future2.isDone());

    }

    /**
     * 在Java8中,CompletableFuture提供了非常強大的Future的擴展功能,可以幫助我們簡化異步編程的復雜性,
     * 并且提供了函數(shù)式編程的能力,可以通過回調(diào)的方式處理計算結(jié)果,也提供了轉(zhuǎn)換和組合 CompletableFuture 的方法
     *
     *  注意: 方法中有Async一般表示另起一個線程,沒有表示用當前線程
     */
    @Test
    public void test01() throws Exception {
        ExecutorService service = Executors.newFixedThreadPool(5);
        /**
         *  supplyAsync用于有返回值的任務,
         *  runAsync則用于沒有返回值的任務
         *  Executor參數(shù)可以手動指定線程池,否則默認ForkJoinPool.commonPool()系統(tǒng)級公共線程池
         */
        CompletableFuture<String> future = CompletableFuture.supplyAsync(() -> {
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            return "xiaoxuzhu";
        }, service);
        CompletableFuture<Void> data = CompletableFuture.runAsync(() -> System.out.println("xiaoxuzhu"));
        /**
         * 計算結(jié)果完成回調(diào)
         */
        future.whenComplete((x,y)-> System.out.println("有延遲3秒:執(zhí)行當前任務的線程繼續(xù)執(zhí)行:"+x+","+y)); //執(zhí)行當前任務的線程繼續(xù)執(zhí)行
        data.whenCompleteAsync((x,y)-> System.out.println("交給線程池另起線程執(zhí)行:"+x+","+y)); // 交給線程池另起線程執(zhí)行
        future.exceptionally(Throwable::toString);
        //System.out.println(future.get());
        /**
         * thenApply,一個線程依賴另一個線程可以使用,出現(xiàn)異常不執(zhí)行
         */
        //第二個線程依賴第一個的結(jié)果
        CompletableFuture<Integer> future1 = CompletableFuture.supplyAsync(() -> 5).thenApply(x -> x);

        /**
         * handle 是執(zhí)行任務完成時對結(jié)果的處理,第一個出現(xiàn)異常繼續(xù)執(zhí)行
         */
        CompletableFuture<Integer> future2 = future1.handleAsync((x, y) -> x + 2);
        System.out.println(future2.get());//7
        /**
         * thenAccept 消費處理結(jié)果,不返回
         */
        future2.thenAccept(System.out::println);
        /**
         * thenRun  不關(guān)心任務的處理結(jié)果。只要上面的任務執(zhí)行完成,就開始執(zhí)行
         */
        future2.thenRunAsync(()-> System.out.println("繼續(xù)下一個任務"));
        /**
         * thenCombine 會把 兩個 CompletionStage 的任務都執(zhí)行完成后,兩個任務的結(jié)果交給 thenCombine 來處理
         */
        CompletableFuture<Integer> future3 = future1.thenCombine(future2, Integer::sum);
        System.out.println(future3.get()); // 5+7=12
        /**
         * thenAcceptBoth : 當兩個CompletionStage都執(zhí)行完成后,把結(jié)果一塊交給thenAcceptBoth來進行消耗
         */
        future1.thenAcceptBothAsync(future2,(x,y)-> System.out.println(x+","+y)); //5,7
        /**
         * applyToEither
         * 兩個CompletionStage,誰執(zhí)行返回的結(jié)果快,我就用那個CompletionStage的結(jié)果進行下一步的轉(zhuǎn)化操作
         */
        CompletableFuture<Integer> future4 = future1.applyToEither(future2, x -> x);
        System.out.println(future4.get()); //5
        /**
         * acceptEither
         * 兩個CompletionStage,誰執(zhí)行返回的結(jié)果快,我就用那個CompletionStage的結(jié)果進行下一步的消耗操作
         */
        future1.acceptEither(future2, System.out::println);
        /**
         * runAfterEither
         * 兩個CompletionStage,任何一個完成了都會執(zhí)行下一步的操作(Runnable
         */
        future1.runAfterEither(future,()-> System.out.println("有一個完成了,我繼續(xù)"));
        /**
         * runAfterBoth
         * 兩個CompletionStage,都完成了計算才會執(zhí)行下一步的操作(Runnable)
         */
        future1.runAfterBoth(future,()-> System.out.println("都完成了,我繼續(xù)"));
        /**
         * thenCompose 方法
         * thenCompose 方法允許你對多個 CompletionStage 進行流水線操作,第一個操作完成時,將其結(jié)果作為參數(shù)傳遞給第二個操作
         * thenApply是接受一個函數(shù),thenCompose是接受一個future實例,更適合處理流操作
         */
        future1.thenComposeAsync(x->CompletableFuture.supplyAsync(()->x+1))
                .thenComposeAsync(x->CompletableFuture.supplyAsync(()->x+2))
                .thenCompose(x->CompletableFuture.runAsync(()-> System.out.println("流操作結(jié)果:"+x)));
        TimeUnit.SECONDS.sleep(5);//主線程sleep,等待其他線程執(zhí)行
    }
}


多學一個知識點

四、推薦專欄

《JAVA從零到壹》

《JAVA筑基100例》

五、示例源碼下載

關(guān)注下面的公眾號,回復筑基+題目號

筑基77

本站僅提供存儲服務,所有內(nèi)容均由用戶發(fā)布,如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請點擊舉報
打開APP,閱讀全文并永久保存 查看更多類似文章
猜你喜歡
類似文章
Java異步編程接口:Callable和Future | Aoho''s Blog
ExecutorService的十個使用技巧
Java 5.0多線程編程
Java 如何實現(xiàn)線程間通信?
多線程下的鎖
兩個例子解釋Callable 和 Future接口
更多類似文章 >>
生活服務
分享 收藏 導長圖 關(guān)注 下載文章
綁定賬號成功
后續(xù)可登錄賬號暢享VIP特權(quán)!
如果VIP功能使用有故障,
可點擊這里聯(lián)系客服!

聯(lián)系客服