欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

java之使用多線(xiàn)程代替for循環(huán)(解決主線(xiàn)程提前結(jié)束問(wèn)題)

 更新時(shí)間:2023年03月10日 08:55:24   作者:大聰明」  
這篇文章主要介紹了java之使用多線(xiàn)程代替for循環(huán)(解決主線(xiàn)程提前結(jié)束問(wèn)題),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

在使用之前先介紹一個(gè)并發(fā)需要用到的方法:

CountDownLatch

  • CountDownLatch(也叫閉鎖)是一個(gè)同步協(xié)助類(lèi),允許一個(gè)或多個(gè)線(xiàn)程等待,直到其他線(xiàn)程完成操作集。
  • CountDownLatch 使用給定的計(jì)數(shù)值(count)初始化。await 方法會(huì)阻塞直到當(dāng)前的計(jì)數(shù)值(count)由于 countDown 方法的調(diào)用達(dá)到 0,count 為 0 之后所有等待的線(xiàn)程都會(huì)被釋放,并且隨后對(duì)await方法的調(diào)用都會(huì)立即返回。

構(gòu)造方法:

//參數(shù)count為計(jì)數(shù)值
public CountDownLatch(int count) {};  

常用方法

// 調(diào)用 await() 方法的線(xiàn)程會(huì)被掛起,它會(huì)等待直到 count 值為 0 才繼續(xù)執(zhí)行
public void await() throws InterruptedException {};
 
// 和 await() 類(lèi)似,若等待 timeout 時(shí)長(zhǎng)后,count 值還是沒(méi)有變?yōu)?0,不再等待,繼續(xù)執(zhí)行
public boolean await(long timeout, TimeUnit unit) throws InterruptedException {};
 
// 會(huì)將 count 減 1,直至為 0
public void countDown() {};

使用案例

  • 首先是創(chuàng)建實(shí)例 CountDownLatch countDown = new CountDownLatch(2);
  • 需要同步的線(xiàn)程執(zhí)行完之后,計(jì)數(shù) -1, countDown.countDown();
  • 需要等待其他線(xiàn)程執(zhí)行完畢之后,再運(yùn)行的線(xiàn)程,調(diào)用 countDown.await()實(shí)現(xiàn)阻塞同步。
  • 如下。

應(yīng)用場(chǎng)景

CountDownLatch 一般用作多線(xiàn)程倒計(jì)時(shí)計(jì)數(shù)器,強(qiáng)制它們等待其他一組(CountDownLatch的初始化決定)任務(wù)執(zhí)行完成。

CountDownLatch的兩種使用場(chǎng)景:

  • 讓多個(gè)線(xiàn)程等待,模擬并發(fā)。
  • 讓單個(gè)線(xiàn)程等待,多個(gè)線(xiàn)程(任務(wù))完成后,進(jìn)行匯總合并。

場(chǎng)景1:模擬并發(fā)

import java.util.concurrent.CountDownLatch;
 
/**
 * 讓多個(gè)線(xiàn)程等待:模擬并發(fā),讓并發(fā)線(xiàn)程一起執(zhí)行
 */
public class CountDownLatchTest {
    public static void main(String[] args) throws InterruptedException {
 
        CountDownLatch countDownLatch = new CountDownLatch(1);
        
        for (int i = 0; i < 5; i++) {
            new Thread(() -> {
                try {
                    // 等待
                    countDownLatch.await();
                    String parter = "【" + Thread.currentThread().getName() + "】";
                    System.out.println(parter + "開(kāi)始執(zhí)行……");
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }).start();
        }
 
        Thread.sleep(2000);
       
        countDownLatch.countDown();
    }
}

場(chǎng)景2:多個(gè)線(xiàn)程完成后,進(jìn)行匯總合并

很多時(shí)候,我們的并發(fā)任務(wù),存在前后依賴(lài)關(guān)系;比如數(shù)據(jù)詳情頁(yè)需要同時(shí)調(diào)用多個(gè)接口獲取數(shù)據(jù),并發(fā)請(qǐng)求獲取到數(shù)據(jù)后、需要進(jìn)行結(jié)果合并;或者多個(gè)數(shù)據(jù)操作完成后,需要數(shù)據(jù) check;這其實(shí)都是:在多個(gè)線(xiàn)程(任務(wù))完成后,進(jìn)行匯總合并的場(chǎng)景。

import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CountDownLatch;
 
/**
 * 讓單個(gè)線(xiàn)程等待:多個(gè)線(xiàn)程(任務(wù))完成后,進(jìn)行匯總合并
 */
public class CountDownLatchTest3 {
 
    //用于聚合所有的統(tǒng)計(jì)指標(biāo)
    private static Map map = new ConcurrentHashMap();
    //創(chuàng)建計(jì)數(shù)器,這里需要統(tǒng)計(jì)4個(gè)指標(biāo)
    private static CountDownLatch countDownLatch = new CountDownLatch(4);
 
    public static void main(String[] args) throws Exception {
 
        //記錄開(kāi)始時(shí)間
        long startTime = System.currentTimeMillis();
 
        Thread countUserThread = new Thread(() -> {
            try {
                System.out.println("正在統(tǒng)計(jì)新增用戶(hù)數(shù)量");
                Thread.sleep(3000);//任務(wù)執(zhí)行需要3秒
                map.put("userNumber", 100);//保存結(jié)果值
                System.out.println("統(tǒng)計(jì)新增用戶(hù)數(shù)量完畢");
                countDownLatch.countDown();//標(biāo)記已經(jīng)完成一個(gè)任務(wù)
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });
        Thread countOrderThread = new Thread(() -> {
            try {
                System.out.println("正在統(tǒng)計(jì)訂單數(shù)量");
                Thread.sleep(3000);//任務(wù)執(zhí)行需要3秒
                map.put("countOrder", 20);//保存結(jié)果值
                System.out.println("統(tǒng)計(jì)訂單數(shù)量完畢");
                countDownLatch.countDown();//標(biāo)記已經(jīng)完成一個(gè)任務(wù)
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });
 
        Thread countGoodsThread = new Thread(() -> {
            try {
                System.out.println("正在商品銷(xiāo)量");
                Thread.sleep(3000);//任務(wù)執(zhí)行需要3秒
                map.put("countGoods", 300);//保存結(jié)果值
                System.out.println("統(tǒng)計(jì)商品銷(xiāo)量完畢");
                countDownLatch.countDown();//標(biāo)記已經(jīng)完成一個(gè)任務(wù)
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });
 
        Thread countmoneyThread = new Thread(() -> {
            try {
                System.out.println("正在總銷(xiāo)售額");
                Thread.sleep(3000);//任務(wù)執(zhí)行需要3秒
                map.put("countMoney", 40000);//保存結(jié)果值
                System.out.println("統(tǒng)計(jì)銷(xiāo)售額完畢");
                countDownLatch.countDown();//標(biāo)記已經(jīng)完成一個(gè)任務(wù)
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });
        
        //啟動(dòng)子線(xiàn)程執(zhí)行任務(wù)
        countUserThread.start();
        countGoodsThread.start();
        countOrderThread.start();
        countmoneyThread.start();
 
        try {
            //主線(xiàn)程等待所有統(tǒng)計(jì)指標(biāo)執(zhí)行完畢
            countDownLatch.await();
            long endTime = System.currentTimeMillis();//記錄結(jié)束時(shí)間
            System.out.println("------統(tǒng)計(jì)指標(biāo)全部完成--------");
            System.out.println("統(tǒng)計(jì)結(jié)果為:" + map);
            System.out.println("任務(wù)總執(zhí)行時(shí)間為" + (endTime - startTime) + "ms");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
 
    }
}

接下來(lái)進(jìn)入正題

使用多線(xiàn)程代替for循環(huán)提高查詢(xún)效率,并且防止主線(xiàn)程提前結(jié)束導(dǎo)致其他線(xiàn)程數(shù)據(jù)錯(cuò)誤

直接上代碼:

@Override
    public AppResponse getLocations() throws InterruptedException {
        List<GetLocationVO> vos = new ArrayList<>();
        vos = projectDao.getLocationOne();    
//      原來(lái)的代碼
//        for (GetLocationVO vo : vos) {
//            List<LocationVO> children = projectDao.getLocationChildren(vo.getId());
//            vo.setChildren(children);
//        }
        //改造后的代碼
        Thread(vos,10);
        return AppResponse.success("查詢(xún)成功",vos);
    }
 
    //此處有加鎖
    public synchronized void Thread(List<GetLocationVO> list, int nThread) throws InterruptedException {
        if (CollectionUtils.isEmpty(list) || nThread <= 0 || CollectionUtils.isEmpty(list)) {
            return;
        }
        CountDownLatch latch = new CountDownLatch(list.size());//創(chuàng)建一個(gè)計(jì)數(shù)器(大小為當(dāng)前數(shù)組的大小,確保所有執(zhí)行完主線(xiàn)程才結(jié)束)
        ExecutorService pool = Executors.newFixedThreadPool(nThread);//創(chuàng)建一個(gè)固定的線(xiàn)程池
        for (GetLocationVO vo : list) {
            pool.execute(() -> {
                //處理的業(yè)務(wù)
                List<LocationVO> children = projectDao.getLocationChildren(vo.getId());
                vo.setChildren(children);
                latch.countDown();
            });
        }
        latch.await();
        pool.shutdown();
    }

總結(jié)

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

相關(guān)文章

  • Java的jmap命令的具體使用

    Java的jmap命令的具體使用

    jmap是JDK提供的一個(gè)可以生成Java虛擬機(jī)的堆轉(zhuǎn)儲(chǔ)快照dump文件的命令行工具,本文主要介紹了Java的jmap命令的具體使用,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-03-03
  • 基于SpringBoot實(shí)現(xiàn)郵箱找回密碼的代碼示例

    基于SpringBoot實(shí)現(xiàn)郵箱找回密碼的代碼示例

    本文主要介紹了如何基于SpringBoot實(shí)現(xiàn)郵箱找回密碼,文中通過(guò)代碼示例給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下
    2024-02-02
  • Java多種方式實(shí)現(xiàn)生產(chǎn)者消費(fèi)者模式

    Java多種方式實(shí)現(xiàn)生產(chǎn)者消費(fèi)者模式

    這篇文章主要介紹了Java多種方式實(shí)現(xiàn)生產(chǎn)者消費(fèi)者模式,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-07-07
  • Java定時(shí)任務(wù)原理詳解

    Java定時(shí)任務(wù)原理詳解

    當(dāng)下,java編碼過(guò)程中,實(shí)現(xiàn)定時(shí)任務(wù)的方式主要以以下兩種為主:spring框架的@Scheduled和quzrtz框架。本文主要就二者的框架原理實(shí)現(xiàn)做一個(gè)入門(mén)引導(dǎo),為了解深層實(shí)現(xiàn)細(xì)節(jié)做一定的鋪墊
    2022-07-07
  • MyBatis中動(dòng)態(tài)SQL語(yǔ)句@Provider的用法

    MyBatis中動(dòng)態(tài)SQL語(yǔ)句@Provider的用法

    本文主要介紹了MyBatis中動(dòng)態(tài)SQL語(yǔ)句@Provider的用法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-06-06
  • 簡(jiǎn)談java并發(fā)FutureTask的實(shí)現(xiàn)

    簡(jiǎn)談java并發(fā)FutureTask的實(shí)現(xiàn)

    這篇文章主要介紹了簡(jiǎn)談java并發(fā)FutureTask的實(shí)現(xiàn),FutureTask都是用于獲取線(xiàn)程執(zhí)行的返回結(jié)果。文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,,需要的朋友可以參考下
    2019-06-06
  • Spring MVC的參數(shù)綁定和返回值問(wèn)題

    Spring MVC的參數(shù)綁定和返回值問(wèn)題

    這篇文章主要介紹了Spring MVC的參數(shù)綁定和返回值問(wèn)題,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-02-02
  • Spring MVC實(shí)現(xiàn)的登錄攔截器代碼分享

    Spring MVC實(shí)現(xiàn)的登錄攔截器代碼分享

    這篇文章主要介紹了Spring MVC實(shí)現(xiàn)的登錄攔截器代碼分享,涉及攔截器的簡(jiǎn)單介紹,攔截器和過(guò)濾器的區(qū)以及攔截器實(shí)現(xiàn)代碼等相關(guān)內(nèi)容,這里分享給大家,供需要的朋友參考。
    2017-10-10
  • 從零實(shí)現(xiàn)一個(gè)簡(jiǎn)單的Spring Bean容器的代碼案例

    從零實(shí)現(xiàn)一個(gè)簡(jiǎn)單的Spring Bean容器的代碼案例

    Spring是一個(gè)非常流行的Java?Web開(kāi)發(fā)框架,它提供了強(qiáng)大的依賴(lài)注入、面向切面編程、聲明式事務(wù)管理等功能,為開(kāi)發(fā)者提供了高效、快速地構(gòu)建Web應(yīng)用程序的工具,在這篇文章中,咱們將一步一步地構(gòu)建一個(gè)簡(jiǎn)單的SpringBean容器,需要的朋友可以參考下
    2023-06-06
  • Spark SQL的自定義函數(shù)UDF使用

    Spark SQL的自定義函數(shù)UDF使用

    Spark Sql可以通過(guò)UDF來(lái)對(duì)DataFrame的Column進(jìn)行自定義操作。在特定場(chǎng)景下定義UDF可能需要用到Spark Context以外的資源或數(shù)據(jù)。比如從List或Map中取值,或是通過(guò)連接池從外部的數(shù)據(jù)源中讀取數(shù)據(jù),然后再參與Column的運(yùn)算
    2023-02-02

最新評(píng)論