你应该这样去开发接口:Java多线程并行计算(Google的Guava使用)

2024-05-27 06:58

本文主要是介绍你应该这样去开发接口:Java多线程并行计算(Google的Guava使用),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

所谓的高并发除了在架构上的高屋建瓴,还得需要开发人员在具体业务开发中注重自己的每一行代码、每一个细节,面子有的同时,更重要的还是要有里子。

面对性能,我们一定要有自己的工匠精神,不可以对任何一行代码妥协!

今天和大家分享在业务开发中如何降低接口响应时间的一个小技巧,也是大家日常开发中比较普遍存在的一个问题,即如何提高程序的并行计算能力?

本文主要包含以下内容:

  1. 顺序执行很慢
  2. 线程池+Future并行计算
  3. 使用Java8的CompletableFuture
  4. 使用Guava的ListenableFuture

本文包含代码内容较多,大家可收藏后自己跟着动手验证一番~

顺序执行

很多时候,我们开发一个接口时候,需要调用多个方法,然后将各个方法返回的数据一起组装返回给前端,比如这样的:

可以看到我这里调用了4个方法,每一个方法为模拟真实耗时,所以都是延迟100ms返回一个字符串:

可想而知,我们这个接口的响应时间一定会超过400ms,多次执行都会在400ms多一点:

耗时:403ms耗时:409ms耗时:406ms

这就是顺序执行,也许大家觉得很Low,但是想想自己的代码很多时候不就是这样子的么?

线程池+Future并行计算

顺序执行确实很慢,所以我们需要并行执行,即同时调用这四个方法,熟悉Java多线程的都知道,每个方法单独开启一个线程异步去执行就好了,等全部执行完了拿到独立线程执行的结果再组装起来就可以了。

但是每次调用都需要创建四个线程,线程的创建和销毁都是需要开销的,所以我们就有了池化技术。

线程池、数据库的连接池等都是采用的池化技术:预先初始生成创建好的线程,等需要调用的时候拿来即用,线程完成工作后回归空闲状态,等待下一次任务的到来,这样就避免了线程频繁的创建、销毁,提高了程序的响应性能。

所以我们在做并行计算的时候一定要充分的利用线程池的相关技术,关于线程池的技术在我的另外一篇文章单独讲到,不了解的同学可以初步了解一下,面试也是必会题之一:

Java线程池基础扫盲

下面我们直接上代码:

线程池+Future

多运行几次,看输出响应时间:

耗时:108ms耗时:105ms耗时:105ms

效果是不是很明显?

直接相当于一个方法的调用耗时,实际开发中如果你的一个接口经过压测耗时在100ms左右(大多数正规公司对接口性能都会要求不超过100ms),那么再通过线程池+Future并行计算的方式,并可以瞬间将你的接口性能提高上去,再也不用担心压测不过了。

有时候测试同学告诉你接口压测不过是不是觉得很没面子?那是对你职业水平很大的否定~

Java8的CompletableFuture

Future是java.util.concurrent并发包中的接口类,用来表示一个线程异步执行后的结果,有如下核心方法:

  • Future.get():阻塞调用线程,直到计算结果返回
  • Future.isDone():判断线程是否执行完毕
  • Future.cancel():取消当前线程的执行

我们可以知道的是,Future.get()是阻塞调用的,要想拿到线程执行的结果,必须是Future.get()阻塞或者while(Future.isDone())轮询方式调用。这种方式叫“主动拉(pull)”,现在都流行响应式编程,即“主动推(push)”的方式,当线程执行完了,你告诉我就好了。

Java8设计了CompletableFuture这样的一个类,我们先来看看如何用CompletableFuture来开发之前的代码:

CompletableFuture并行计算

这里可以看到实现方式和Future并没有什么不同,但是CompletableFuture提供了很多方便的方法,比如代码中的allOf,thenApplyAsync,可以将多个CompletableFuture组合成一个CompletableFuture,最后调用join方法阻塞拿到结果。多次调用该接口耗时如下:

耗时:110ms耗时:108ms耗时:105ms

CompletableFuture类中有很多的方法(50+)可以供大家使用,不像Future只要那么几个方法可以使用,这也是Java自有库对Future的一个增强。

这里只是简单展示了CompletableFuture的一种用法,实际开发中大家需要根据不同的场景去选择使用不同的方法,这里对API不做具体介绍了。

Guava的ListenableFuture

总是有一些牛逼的公司牛逼的人出一些牛逼的开源组件要比官方自带的工具类要好得多,同样,谷歌开源的Guava中的ListenableFuture接口对java自带的Future接口做了进一步拓展,并且提供了静态工具类Futures。

针对上面的代码,我们看如何使用ListenableFuture来实现(与之前不同的是,Guava中需要对线程池再进行一次包装):

执行三次请求耗时:

耗时:103ms耗时:101ms耗时:103ms

最后

以上就是如何让自己的接口并行计算起来的三种实现方式,属于日常开发中比较常用的一个小技巧,这里没有过多说明这三种方式的具体区别,实际上还需要大家不断的在开发中去使用,查阅更多相关源码和资料,只有等你真正用起来的时候,你才能有所体会!

这篇关于你应该这样去开发接口:Java多线程并行计算(Google的Guava使用)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1006761

相关文章

Spring Boot + MyBatis Plus 高效开发实战从入门到进阶优化(推荐)

《SpringBoot+MyBatisPlus高效开发实战从入门到进阶优化(推荐)》本文将详细介绍SpringBoot+MyBatisPlus的完整开发流程,并深入剖析分页查询、批量操作、动... 目录Spring Boot + MyBATis Plus 高效开发实战:从入门到进阶优化1. MyBatis

SpringCloud动态配置注解@RefreshScope与@Component的深度解析

《SpringCloud动态配置注解@RefreshScope与@Component的深度解析》在现代微服务架构中,动态配置管理是一个关键需求,本文将为大家介绍SpringCloud中相关的注解@Re... 目录引言1. @RefreshScope 的作用与原理1.1 什么是 @RefreshScope1.

Java并发编程必备之Synchronized关键字深入解析

《Java并发编程必备之Synchronized关键字深入解析》本文我们深入探索了Java中的Synchronized关键字,包括其互斥性和可重入性的特性,文章详细介绍了Synchronized的三种... 目录一、前言二、Synchronized关键字2.1 Synchronized的特性1. 互斥2.

Spring Boot 配置文件之类型、加载顺序与最佳实践记录

《SpringBoot配置文件之类型、加载顺序与最佳实践记录》SpringBoot的配置文件是灵活且强大的工具,通过合理的配置管理,可以让应用开发和部署更加高效,无论是简单的属性配置,还是复杂... 目录Spring Boot 配置文件详解一、Spring Boot 配置文件类型1.1 applicatio

Python基于wxPython和FFmpeg开发一个视频标签工具

《Python基于wxPython和FFmpeg开发一个视频标签工具》在当今数字媒体时代,视频内容的管理和标记变得越来越重要,无论是研究人员需要对实验视频进行时间点标记,还是个人用户希望对家庭视频进行... 目录引言1. 应用概述2. 技术栈分析2.1 核心库和模块2.2 wxpython作为GUI选择的优

使用Sentinel自定义返回和实现区分来源方式

《使用Sentinel自定义返回和实现区分来源方式》:本文主要介绍使用Sentinel自定义返回和实现区分来源方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Sentinel自定义返回和实现区分来源1. 自定义错误返回2. 实现区分来源总结Sentinel自定

Pandas使用SQLite3实战

《Pandas使用SQLite3实战》本文主要介绍了Pandas使用SQLite3实战,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录1 环境准备2 从 SQLite3VlfrWQzgt 读取数据到 DataFrame基础用法:读

JSON Web Token在登陆中的使用过程

《JSONWebToken在登陆中的使用过程》:本文主要介绍JSONWebToken在登陆中的使用过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录JWT 介绍微服务架构中的 JWT 使用结合微服务网关的 JWT 验证1. 用户登录,生成 JWT2. 自定义过滤

Java中StopWatch的使用示例详解

《Java中StopWatch的使用示例详解》stopWatch是org.springframework.util包下的一个工具类,使用它可直观的输出代码执行耗时,以及执行时间百分比,这篇文章主要介绍... 目录stopWatch 是org.springframework.util 包下的一个工具类,使用它

Java进行文件格式校验的方案详解

《Java进行文件格式校验的方案详解》这篇文章主要为大家详细介绍了Java中进行文件格式校验的相关方案,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、背景异常现象原因排查用户的无心之过二、解决方案Magandroidic Number判断主流检测库对比Tika的使用区分zip