我的编程空间,编程开发者的网络收藏夹
学习永远不晚

java怎么使用多线程解决主线程提前结束问题

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

java怎么使用多线程解决主线程提前结束问题

这篇文章主要介绍“java怎么使用多线程解决主线程提前结束问题”,在日常操作中,相信很多人在java怎么使用多线程解决主线程提前结束问题问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”java怎么使用多线程解决主线程提前结束问题”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

CountDownLatch

  • CountDownLatch(也叫闭锁)是一个同步协助类,允许一个或多个线程等待,直到其他线程完成操作集。

  • CountDownLatch 使用给定的计数值(count)初始化。await 方法会阻塞直到当前的计数值(count)由于 countDown 方法的调用达到 0,count 为 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,直至为 0public void countDown() {};

使用案例

  • 首先是创建实例 CountDownLatch countDown = new CountDownLatch(2);

  • 需要同步的线程执行完之后,计数 -1, countDown.countDown();

  • 需要等待其他线程执行完毕之后,再运行的线程,调用 countDown.await()实现阻塞同步。

  • 如下。

应用场景

CountDownLatch 一般用作多线程倒计时计数器,强制它们等待其他一组(CountDownLatch的初始化决定)任务执行完成。

CountDownLatch的两种使用场景:

  • 让多个线程等待,模拟并发。

  • 让单个线程等待,多个线程(任务)完成后,进行汇总合并。

场景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:多个线程完成后,进行汇总合并

很多时候,我们的并发任务,存在前后依赖关系;比如数据详情页需要同时调用多个接口获取数据,并发请求获取到数据后、需要进行结果合并;或者多个数据操作完成后,需要数据 check;这其实都是:在多个线程(任务)完成后,进行汇总合并的场景。

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怎么使用多线程解决主线程提前结束问题”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注编程网网站,小编会继续努力为大家带来更多实用的文章!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

java怎么使用多线程解决主线程提前结束问题

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

java怎么使用多线程解决主线程提前结束问题

这篇文章主要介绍“java怎么使用多线程解决主线程提前结束问题”,在日常操作中,相信很多人在java怎么使用多线程解决主线程提前结束问题问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”java怎么使用多线程解决
2023-07-05

java之使用多线程代替for循环(解决主线程提前结束问题)

这篇文章主要介绍了java之使用多线程代替for循环(解决主线程提前结束问题),具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
2023-03-10

Java多线程之线程安全问题怎么解决

本篇内容主要讲解“Java多线程之线程安全问题怎么解决”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Java多线程之线程安全问题怎么解决”吧!1.线程安全概述1.1什么是线程安全问题首先我们需要
2023-06-30

Java多线程死锁问题怎么解决

解决Java多线程死锁问题的常用方法有以下几种:1. 避免使用多个锁:尽量减少使用多个锁来降低出现死锁的概率。2. 按照固定的顺序获取锁:对于多个锁的获取,确保线程按照固定的顺序获取锁,避免出现循环等待的情况。3. 设置超时时间:在获取锁的
2023-09-22

MFC多线程传参问题怎么解决

本篇内容主要讲解“MFC多线程传参问题怎么解决”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“MFC多线程传参问题怎么解决”吧!在一个程序中,这些独立运行的程序片断叫作“线程”(Thread),利
2023-06-17

Java子线程任务异常和主线程事务回滚问题怎么解决

这篇文章主要介绍了Java子线程任务异常和主线程事务回滚问题怎么解决的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Java子线程任务异常和主线程事务回滚问题怎么解决文章都会有所收获,下面我们一起来看看吧。一、提
2023-06-30

java treemap线程安全问题怎么解决

要解决Java TreeMap的线程安全问题,有以下几种方法:1. 使用Collections.synchronizedMap()方法包装TreeMap对象,将其转换为线程安全的Map对象。示例代码如下:Map synchronizedMa
2023-10-20

java for循环内执行多线程问题怎么解决

这篇文章主要介绍了java for循环内执行多线程问题怎么解决的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇java for循环内执行多线程问题怎么解决文章都会有所收获,下面我们一起来看看吧。java用多线程来
2023-07-05

Python互斥锁怎么解决多线程问题

这篇文章给大家分享的是有关Python互斥锁怎么解决多线程问题的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。python主要应用领域有哪些1、云计算,典型应用OpenStack。2、WEB前端开发,众多大型网站均
2023-06-14

springboot多线程高并发问题怎么解决

要解决Spring Boot多线程高并发问题,你可以考虑以下几个方面:1. 线程池配置:合理配置线程池的大小、队列容量等参数,以满足高并发场景下的需求。可以使用Spring Boot提供的线程池配置方式,如通过在application.pr
2023-08-24

python多线程效率低问题怎么解决

Python中的多线程效率低的问题主要是由于Python的全局解释器锁(GIL)导致的。GIL是Python解释器中的一个机制,它确保同一时刻只有一个线程执行Python字节码。要解决Python多线程效率低的问题,可以考虑以下几种方法:1
2023-09-04

Linux多线程及多线程并发访问同一块内存的问题怎么解决

这篇文章主要介绍了Linux多线程及多线程并发访问同一块内存的问题怎么解决的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Linux多线程及多线程并发访问同一块内存的问题怎么解决文章都会有所收获,下面我们一起来看
2023-07-05

Java SimpleDateFormat线程不安全问题怎么解决

本篇内容主要讲解“Java SimpleDateFormat线程不安全问题怎么解决”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Java SimpleDateFormat线程不安全问题怎么解决”
2023-07-05

java多线程写文件出现冲突问题怎么解决

在多线程写文件时,可能会出现冲突问题,可以通过以下几种方式解决:使用锁(Lock):在文件写入操作之前,先获取一个锁,其他线程需要等待锁释放后才能进行写入操作。使用同步块(synchronized):使用同步块来确保多线程写文件时的互斥性,
java多线程写文件出现冲突问题怎么解决
2024-02-29

spring scheduled单线程和多线程使用的坑怎么解决

本篇内容介绍了“spring scheduled单线程和多线程使用的坑怎么解决”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!公司在使用定时任
2023-06-29

Java线程技术中的死锁问题怎么解决

这篇文章主要介绍“Java线程技术中的死锁问题怎么解决”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Java线程技术中的死锁问题怎么解决”文章能帮助大家解决问题。我们知道,使用 synchroniz
2023-06-02

Java调度线程池ScheduledThreadPoolExecutor不执行问题怎么解决

本篇内容主要讲解“Java调度线程池ScheduledThreadPoolExecutor不执行问题怎么解决”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Java调度线程池ScheduledTh
2023-07-05

java单例模式和线程安全问题怎么解决

这篇文章主要介绍“java单例模式和线程安全问题怎么解决”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“java单例模式和线程安全问题怎么解决”文章能帮助大家解决问题。单例模式、多实例模式、和线程安全
2023-07-05

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录