ホームページ  >  記事  >  Java  >  Java でマルチスレッドを使用して、メインスレッドが途中で終了しないようにするにはどうすればよいですか?

Java でマルチスレッドを使用して、メインスレッドが途中で終了しないようにするにはどうすればよいですか?

PHPz
PHPz転載
2023-04-27 16:40:151154ブラウズ

CountDownLatch

  • CountDownLatch (ロックとも呼ばれます) は、他のスレッドが一連の操作を完了するまで 1 つ以上のスレッドを待機できるようにする同期ヘルパー クラスです。 。

  • CountDownLatch 指定されたカウント値で初期化されます。 await メソッドは現在のカウント値 (count) になるまでブロックされます。countDown メソッドの呼び出しが 0 に達するため、カウントが 0 になると待機中のスレッドはすべて解放され、それ以降の await メソッドの呼び出しはすぐに戻ります。

構築方法:

//参数count为计数值
public CountDownLatch(int count) {};

一般的な方法

// 调用 await() 方法的线程会被挂起,它会等待直到 count 值为 0 才继续执行
public void await() throws InterruptedException {};
 
// 和 await() 类似,若等待 timeout 时长后,count 值还是没有变为 0,不再等待,继续执行
public boolean await(long timeout, TimeUnit unit) throws InterruptedException {};
 
// 会将 count 减 1,直至为 0
public void countDown() {};

使用例

  • 最初は、インスタンス CountDownLatch countDown = new CountDownLatch(2);

  • 同期する必要があるスレッドが実行された後、カウントは -1、countDown.countDown();

  • 実行前に他のスレッドの実行が完了するのを待つ必要があるスレッドの場合は、countDown.await() を呼び出してブロック同期を実現します。 ############次のように。

  • アプリケーション シナリオ

  • CountDownLatch は通常、マルチスレッドのカウントダウン カウンターとして使用され、別のタスク セット (初期化) の実行が完了するまで待機させます。 CountDownLatch の決定)。

CountDownLatch の 2 つの使用シナリオ:

複数のスレッドを待機させ、同時実行をシミュレートします。

  • 単一スレッドを待機させ、複数のスレッド (タスク) が完了したら、要約してマージします。

  • シナリオ 1: 同時実行性のシミュレーション

    import java.util.concurrent.CountDownLatch;
     
    /**
     * 让多个线程等待:模拟并发,让并发线程一起执行
     */
    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 + "开始执行……");
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }).start();
            }
     
            Thread.sleep(2000);
           
            countDownLatch.countDown();
        }
    }
  • シナリオ 2: 複数のスレッドが完了した後、要約してマージする

何度も、同時タスクが存在します。依存関係の前後です。たとえば、データの詳細ページでは、データを取得するために複数のインターフェイスを同時に呼び出す必要があります。同時リクエストがデータを取得した後、結果をマージする必要があります。または、複数のデータ操作が完了した後、データ チェックが行われます。は必須です; これは実際には次のとおりです: 複数のスレッド (タスク) が完了した後、シナリオが要約され、マージされます。

import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CountDownLatch;
 
/**
 * 让单个线程等待:多个线程(任务)完成后,进行汇总合并
 */
public class CountDownLatchTest3 {
 
    //用于聚合所有的统计指标
    private static Map map = new ConcurrentHashMap();
    //创建计数器,这里需要统计4个指标
    private static CountDownLatch countDownLatch = new CountDownLatch(4);
 
    public static void main(String[] args) throws Exception {
 
        //记录开始时间
        long startTime = System.currentTimeMillis();
 
        Thread countUserThread = new Thread(() -> {
            try {
                System.out.println("正在统计新增用户数量");
                Thread.sleep(3000);//任务执行需要3秒
                map.put("userNumber", 100);//保存结果值
                System.out.println("统计新增用户数量完毕");
                countDownLatch.countDown();//标记已经完成一个任务
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });
        Thread countOrderThread = new Thread(() -> {
            try {
                System.out.println("正在统计订单数量");
                Thread.sleep(3000);//任务执行需要3秒
                map.put("countOrder", 20);//保存结果值
                System.out.println("统计订单数量完毕");
                countDownLatch.countDown();//标记已经完成一个任务
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });
 
        Thread countGoodsThread = new Thread(() -> {
            try {
                System.out.println("正在商品销量");
                Thread.sleep(3000);//任务执行需要3秒
                map.put("countGoods", 300);//保存结果值
                System.out.println("统计商品销量完毕");
                countDownLatch.countDown();//标记已经完成一个任务
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });
 
        Thread countmoneyThread = new Thread(() -> {
            try {
                System.out.println("正在总销售额");
                Thread.sleep(3000);//任务执行需要3秒
                map.put("countMoney", 40000);//保存结果值
                System.out.println("统计销售额完毕");
                countDownLatch.countDown();//标记已经完成一个任务
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });
        
        //启动子线程执行任务
        countUserThread.start();
        countGoodsThread.start();
        countOrderThread.start();
        countmoneyThread.start();
 
        try {
            //主线程等待所有统计指标执行完毕
            countDownLatch.await();
            long endTime = System.currentTimeMillis();//记录结束时间
            System.out.println("------统计指标全部完成--------");
            System.out.println("统计结果为:" + map);
            System.out.println("任务总执行时间为" + (endTime - startTime) + "ms");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
 
    }
}

本題に入りましょう

for ループの代わりにマルチスレッドを使用してクエリ効率を向上させ、メインスレッドが早期に終了して他のスレッドでデータ エラーが発生するのを防ぎます

コードに直接移動します:

@Override
    public AppResponse getLocations() throws InterruptedException {
        List<GetLocationVO> vos = new ArrayList<>();
        vos = projectDao.getLocationOne();    
//      原来的代码
//        for (GetLocationVO vo : vos) {
//            List<LocationVO> children = projectDao.getLocationChildren(vo.getId());
//            vo.setChildren(children);
//        }
        //改造后的代码
        Thread(vos,10);
        return AppResponse.success("查询成功",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());//创建一个计数器(大小为当前数组的大小,确保所有执行完主线程才结束)
        ExecutorService pool = Executors.newFixedThreadPool(nThread);//创建一个固定的线程池
        for (GetLocationVO vo : list) {
            pool.execute(() -> {
                //处理的业务
                List<LocationVO> children = projectDao.getLocationChildren(vo.getId());
                vo.setChildren(children);
                latch.countDown();
            });
        }
        latch.await();
        pool.shutdown();
    }

以上がJava でマルチスレッドを使用して、メインスレッドが途中で終了しないようにするにはどうすればよいですか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事はyisu.comで複製されています。侵害がある場合は、admin@php.cn までご連絡ください。