你应该这样去开发接口:Java多线程并行计算
所谓的高并发除了在架构上的高屋建瓴,还得需要开发人员在具体业务开发中注重自己的每一行代码、每一个细节,面子有的同时,更重要的还是要有里子。
面对性能,我们一定要有自己的工匠精神,不可以对任何一行代码妥协!
今天和大家分享在业务开发中如何降低接口响应时间的一个小技巧,也是大家日常开发中比较普遍存在的一个问题,即如何提高程序的并行计算能力?
本文主要包含以下内容:
- 顺序执行很慢
- 线程池+Future并行计算
- 使用Java8的CompletableFuture
- 使用Guava的ListenableFuture
本文包含代码内容较多,大家可收藏后自己跟着动手验证一番~
顺序执行
很多时候,我们开发一个接口时候,需要调用多个方法,然后将各个方法返回的数据一起组装返回给前端,比如这样的:
可以看到我这里调用了4个方法,每一个方法为模拟真实耗时,所以都是延迟100ms返回一个字符串:
可想而知,我们这个接口的响应时间一定会超过400ms,多次执行都会在400ms多一点:
耗时:403ms耗时:409ms耗时:406ms
这就是顺序执行,也许大家觉得很Low,但是想想自己的代码很多时候不就是这样子的么?
线程池+Future并行计算
顺序执行确实很慢,所以我们需要并行执行,即同时调用这四个方法,熟悉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多线程并行计算的更多相关文章
- 通用权限管理系统多语言开发接口 - java,php 调用接口程序,多业务子系统集成
1:公司里有多个业务系统,需要进行统一重构,有PHP的.有Java的.有.NET的,甚至还有delphi的. 2:公司里有多个数据库系统,有mysql的.有sqlserver的.还有oracel的,甚 ...
- 编程开发之--java多线程学习总结(1)问题引入与概念叙述
1.经典问题,火车站售票,公共票源箱,多个窗口同时取箱中车票销售 package com.lfy.ThreadsSynchronize; /** * 解决办法分析:即我们不能同时让超过两个以上的线程进 ...
- 编程开发之--java多线程学习总结(5)
4.对继承自Runnable的线程进行锁机制的使用 package com.lfy.ThreadsSynchronize; import java.util.concurrent.locks.Lock ...
- 编程开发之--java多线程学习总结(4)
3.使用锁机制lock,unlock package com.lfy.ThreadsSynchronize; import java.util.concurrent.locks.Lock; impor ...
- 编程开发之--java多线程学习总结(2)同步代码块
1.第一种解决办法:同步代码块,关键字synchronized package com.lfy.ThreadsSynchronize; /** * 1.使用同步代码块 * 语法: synchroniz ...
- 编程开发之--java多线程学习总结(6)
5.测试 package com.lfy.ThreadsSynchronize; public class Test { public static void main(String[] args) ...
- 编程开发之--java多线程学习总结(3)类锁
2.使用方法同步 package com.lfy.ThreadsSynchronize; /** * 1.使用同步方法 * 语法:即用 synchronized 关键字修饰方法(注意是在1个对象中用锁 ...
- Java多线程编程模式实战指南(二):Immutable Object模式--转载
本文由本人首次发布在infoq中文站上:http://www.infoq.com/cn/articles/java-multithreaded-programming-mode-immutable-o ...
- Java多线程——Condition条件
简介 Condition中的await()方法相当于Object的wait()方法,Condition中的signal()方法相当于Object的notify()方法,Condition中的signa ...
随机推荐
- 【目录】Java项目开发中的知识记录
此篇文章为学习Java的目录,<a href="#"></>这种的是还没有写的文章.已经加a标签的是已经写完的.没写的文章急切需要的话可以直接留言,不是特别 ...
- LinkedList作为栈和队列的使用
最近在LeekCode用java写一些算法时,经常遇到要使用栈和队列结构,使用栈的话,Stack已经不被推荐使用了,所以栈和队列我们通常都是用LinkedList这种双链表结构实现.Linkedlis ...
- raft协议
一.Raft一致性算法 Eureka:Peer To Peer,每个节点的地位都是均等的,每个节点都可以接收写入请求,每个节点接收请求之后,进行请求打包处理,异步化延迟一点时间,将数据同步给 Eure ...
- Java并发编程(二)如何保证线程同时/交替执行
第一篇文章中,我用如何保证线程顺序执行的例子作为Java并发系列的开胃菜.本篇我们依然不会有源码分析,而是用另外两个多线程的例子来引出Java.util.concurrent中的几个并发工具的用法. ...
- Jmeter(一) - 从入门到精通 - 环境搭建(详解教程)
1.JMeter 介绍 Apache JMeter是100%纯JAVA桌面应用程序,被设计为用于测试客户端/服务端结构的软件(例如web应用程序).它可以用来测试静态和动态资源的性能,例如:静态文件, ...
- Java Arrays.sort()重写comparator方法
先看一下接口 Arrays.sort(T[],Comparator<? super T> c); comparator要重写compare方法 compare方法大概长这样,返回值> ...
- 折腾gcc/g++链接时.o文件及库的顺序问题
gcc/g++链接时.o文件以及库的顺序问题 1 写在前面 最近换了xubuntu12.4,把原来的项目co出来编译的时候报"undefined reference to".猜测是 ...
- Linux——定时清空日志内容和删除日志文件
前言 最近在做性能压测试,会生成大量的日志,导致后续越压越慢,最终磁盘空间占满之类的问题.老是要手动删除日志文件,为避免此类问题发生,编写一个Linux日志定时清理的脚本,一劳永逸. 1.shell脚 ...
- NFS PersistentVolume(11)
一.部署nfs服务端 1.需在 k8s-master 节点上搭建了一个 NFS 服务器,目录为 /nfsdata: yum install -y nfs-utils rpcbind vim /etc/ ...
- 012.Kubernetes的configmap和secret配置
使用configmap对多有的配置文件进行统一管理 一 configmap配置管理 1.1 检查mysql的配置 [root@docker-server1 storage]# kubectl get ...