[实践总结] 典型的串行任务局部并行化处理案例:多任务并发获取航班信息

假设你有一个APP,主要用于查询航班信息,你的APP是没有这些实时数据的,当用户发起查询请求时,你需要到各大航空公司的接口获取信息,最后统一整理加工返回到APP客户端。当然JDK自带了很多高级工具,比如CountDownLatchCyclicBarrier等都可以完成类似的功能,但是仅就我们目前所学的知识,使用join方法即可完成下面的功能。

该例子是典型的串行任务局部并行化处理,用户在APP客户端输入出发地“北京”和目的地“上海”,服务器接收到这个请求之后,先来验证用户的信息,然后到各大航空公司的接口查询信息,最后经过整理加工返回给客户端,每一个航空公司的接口不会都一样,获取的数据格式也不一样,查询的速度也存在着差异,如果再跟航空公司进行串行化交互(逐个地查询),很明显客户端需要等待很长的时间,这样的话,用户体验就会非常差。如果我们将每一个航空公司的查询都交给一个线程去工作,然后在它们结束工作之后统一对数据进行整理,这样就可以极大地节约时间,从而提高用户体验效果。

// 面相接口编程,定义一个查询接口FightQuery
public interface FightQuery {List<String> getRes();
}----------------------------------------------------------------------------------------
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ThreadLocalRandom;
import java.util.concurrent.TimeUnit;// 查询Fight的task,其实就是一个线程的子类,主要用于到各大航空公司获取数据
public class FightQueryTask extends Thread implements FightQuery {private final String origin;private final String destination;private final List<String> flightMsgs = new ArrayList<>();/**** @param airline 航空公司* @param origin 始发站* @param destination 目的地*/public FightQueryTask(String airline, String origin, String destination) {super("[" + airline + "]"); // 学到了这里居然是给线程起名字this.origin = origin;this.destination = destination;}@Overridepublic void run() {System.out.println(Thread.currentThread().getName() + "Query from " + origin + " to " + destination);// 模拟业务逻辑处理int randomVal = ThreadLocalRandom.current().nextInt(10);shortSleep(randomVal);flightMsgs.add(getName() + "-" + randomVal);System.out.println(getName() + "query done");}@Overridepublic List<String> getRes() {return this.flightMsgs;}private static void shortSleep(long time) {try {TimeUnit.SECONDS.sleep(time);} catch (InterruptedException e) {e.printStackTrace();}}
}
----------------------------------------------------------------------------------------
public class FightQueryMain {public static void main(String[] args) {// 实现一下从SH(上海)到北京(BJ)的航班查询List<String> results = flightSearch("SH", "BJ");System.out.println("===========result===========");results.forEach(System.out::println);}private static List<String> flightSearch(String original, String dest) {//①合作的各大航空公司List<String> fightCompanies = Arrays.asList("CSA", "CEA", "HNA");//②创建查询航班信息的线程列表List<FightQueryTask> queryTasks = fightCompanies.stream().map(fightCompany -> new FightQueryTask(fightCompany, original, dest)).toList();//③分别启动这几个线程queryTasks.forEach(Thread::start);//④分别调用每一个线程的join方法,阻塞当前线程queryTasks.forEach(t -> {try {t.join();} catch (InterruptedException e) {throw new RuntimeException(e);}});//⑤当前线程会阻塞住,直到获取每一个查询线程的结果return queryTasks.stream().map(FightQuery::getRes).flatMap(Collection::stream).toList();}
}[CSA]Query from SH to BJ
[HNA]Query from SH to BJ
[CEA]Query from SH to BJ
[HNA]query done
[CEA]query done
[CSA]query done
===========result===========
[CSA]-5
[CEA]-3
[HNA]-0

-----------------------------------------------------------------------------读书笔记摘自书名:Java高并发编程详解:多线程与架构设计 作者:汪文君

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/227579.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Qt】报错error:undefined reference to `vtable for Consumer‘的解决方法

1. 问题原因 在创建完程序后&#xff0c;点击构建&#xff0c;显示编译错误。 错误问题如下: error: undefined reference to vtable在编译输出中查看显示如下&#xff1a; error:undefined reference to vtable for custom2. 原因分析 这个错误通常是因为 C 的虚函数表&am…

《面向机器学习的数据标注规程》摘录

说明&#xff1a;本文使用的标准是2019年的团体标准&#xff0c;最新的国家标准已在2023年发布。 3 术语和定义 3.2 标签 label 标识数据的特征、类别和属性等。 3.4 数据标注员 data labeler 对待标注数据进行整理、纠错、标记和批注等操作的工作人员。 【批注】按照定义…

Java中的网络通信协议与通信模型分析

一、引言 网络通信在现代社会中扮演着重要的角色&#xff0c;而Java作为一种广泛应用于网络编程的编程语言&#xff0c;其网络通信协议与通信模型的分析显得尤为重要。本文将分析Java中常用的网络通信协议和通信模型&#xff0c;探讨其特点和应用场景。 二、网络通信协议 1. …

【已解决】ModuleNotFoundError: No module named ‘tensorflow‘

问题描述 Traceback (most recent call last): File "dataset_tool.py", line 16, in <module> import tensorflow as tf ModuleNotFoundError: No module named tensorflow 如果直接pip install tensorflow&#xff0c;还会报错 解决办法 方法一 pip i…

redis未授权漏洞复现

什么是redis redis就是个数据库&#xff0c;跟mysql不同的地方在于redis主要将数据存在内存中&#xff0c;读写速度非常快 redis未授权 其原因很简单&#xff0c;就是redis服务器在默认安装好不配置的情况下可以直接免密码登录&#xff0c;登录后在web目录写入一句话木马&am…

钉钉中预览打印PDF问题(无法使用blob地址)

使用pdfjs-dist预览文件 依赖 npm install pdfjs-dist2.14.305组件 <template><div id"pageContainer"><div id"viewer"></div></div> </template><script> import pdfjs-dist/web/pdf_viewer.css; import *…

前端设计模式之旅:命令模式

引言 使用命令模式&#xff0c;我们可以将执行特定任务的对象与调用该方法的对象解耦。 核心思想 命令模式的核心思想是将请求封装成一个对象&#xff0c;从而使请求的发起者和请求的执行者解耦。 请求的发起者只需要知道如何创建命令对象并将其传递给请求者&#xff0c;而不需…

极坐标下的牛拉法潮流计算57节点MATLAB程序

微❤关注“电气仔推送”获得资料&#xff08;专享优惠&#xff09; 潮流计算&#xff1a; 潮流计算是根据给定的电网结构、参数和发电机、负荷等元件的运行条件&#xff0c;确定电力系统各部分稳态运行状态参数的计算。通常给定的运行条件有系统中各电源和负荷点的功率、枢纽…

贪心算法:买卖股票的最佳时机II 跳跃游戏 跳跃游戏II

122.买卖股票的最佳时机II 思路&#xff1a; 想要获得利润&#xff0c;至少要以两天为一个交易单元&#xff0c;因为两天才会有股价差。因此可以将最终利润进行分解&#xff0c;如prices[3] - prices[0] (prices[3] - prices[2]) (prices[2] - prices[1]) (prices[1] - pr…

【Mars3d-ModelEntity】实现gltf模型不随地图缩放而改变大小

需求场景&#xff1a; 1.实现gltf模型不随地图缩放而改变大小 相关代码&#xff1a; const graphic new mars3d.graphic.ModelEntity({ name: "警车", position: [116.346929, 30.861947, 401.34], style: { url: "//data.mars3d.cn/gltf/mars/jingche/jingc…

python界面开发,使用wxpython库

入门学习Python时&#xff0c;使从接触一个项目开始&#xff0c;当时需要我开发一个界面&#xff0c;当时综合考量之后&#xff0c;最终选择了今天要分享的内容部分&#xff0c;也就是使用Python来开发&#xff0c;主要使用到的是Python库——wxPython库来进行界面开发&#xf…

Git批量删除本地分支

开发一段时间后&#xff0c;我们本地会有很多无用的分支&#xff0c;一个一个的 git branch -D branchName 又感觉太费时间了&#xff0c;如果要批量删除&#xff0c;可以用下面的命令&#xff1a; git checkout master git branch | grep -v master | xargs git branch -D 注…

论文阅读:Lidar Annotation Is All You Need

目录 概要 Motivation 整体架构流程 技术细节 小结 概要 论文重点在探讨利用点云的地面分割任务作为标注&#xff0c;直接训练Camera的精细2D分割。在以往的地面分割任务中&#xff0c;利用Lidar来做地面分割是目前采用激光雷达方案进行自动驾驶的常见手段。来自Evocargo …

phpstudy是什么?

PHPStudy 是一个集成环境工具&#xff0c;它将 PHP 开发所需的软件&#xff0c;如 Apache&#xff08;Web服务器&#xff09;、MySQL&#xff08;数据库服务器&#xff09;、PHP&#xff08;脚本语言&#xff09;等打包在一起&#xff0c;以便用户能够轻松安装和配置这些软件&a…

UniGUI 之UniDBGrid

目录 1]DataSource设置 2]显示MEMO类型里的文字 3]显示悬浮提示 4]显示当前记录及总记录数 5]读取所有记录&#xff0c;及分页 6]在前面加上序号列 7]不显示标题栏 8]列排序 9]编辑 和 更新 数据 10]获得某单元格里的内容 11]标题别名 12]将某列设置为CheckBox格式 13]列标题…

Redis设计与实现之字符串哈希表列表

目录 一、字符串 1、字符串编码 2、编码的选择 二、哈希表 1、字典编码的哈希表 2、压缩列表编码的哈希表 3、编码的选择 4、哈希命令的实现 三、列表 1、 编码的选择 2、 列表命令的实现 3、阻塞的条件 4、 阻塞 5、 阻塞因 LPUSH 、RPUSH 、LINSERT 等添加命令而…

【C语言】操作符详解(五)

目录 操作符的属性&#xff1a;优先级&#xff0c;结合性 优先级 结合性 表达式求值 整形提升 算术转换 问题表达式解析 表达式1 表达式2 表达式3 总结 操作符的属性&#xff1a;优先级&#xff0c;结合性 优先级 ⭐优先级&#xff1a;优先级指的是&#xff0c;如果一…

网络安全——Iptables防DDoS攻击实验

一、实验目的要求&#xff1a; 二、实验设备与环境&#xff1a; 三、实验原理&#xff1a; 四、实验步骤&#xff1a; 五、实验现象、结果记录及整理&#xff1a; 六、分析讨论与思考题解答&#xff1a; 一、实验目的要求&#xff1a; 1、掌握常见DDoS攻击SYN Flood的攻击…

PCL点云处理之主成分分析(PCA)拟合圆柱参数(C++详细介绍)(二百二十六)

PCA点云处理之主成分分析拟合圆柱参数(C++详细介绍)(二百二十六) 一、算法介绍二、算法设计三、算法实现1.代码2.结果3.矩阵分解的特征值和特征向量的说明一、算法介绍 这个算法的作用是对给定的点云数据进行圆柱拟合,从而得到圆柱的主轴向量、中心点和半径参数。具体来说…

Cmake基础(4)

这篇文章在上一篇的基础之上应用多文件&#xff0c;即一个项目中添加多个文件 文章目录 GLOBsource_group排除文件 上一篇文章的cmake基本不变&#xff0c;这篇文章的重点在于add_executable(${EXECUTABLE_NAME} main.cpp) GLOB file(GLOB cpp_list ${CMAKE_CURRENT_SOURCE_…