网站首页 > 精选教程 正文
所谓的高并发除了在架构上的高屋建瓴,还得需要开发人员在具体业务开发中注重自己的每一行代码、每一个细节,面子有的同时,更重要的还是要有里子。
面对性能,我们一定要有自己的工匠精神,不可以对任何一行代码妥协!
今天和大家分享在业务开发中如何降低接口响应时间的一个小技巧,也是大家日常开发中比较普遍存在的一个问题,即如何提高程序的并行计算能力?
本文主要包含以下内容:
- 顺序执行很慢
- 线程池+Future并行计算
- 使用Java8的CompletableFuture
- 使用Guava的ListenableFuture
本文包含代码内容较多,大家可收藏后自己跟着动手验证一番~
顺序执行
很多时候,我们开发一个接口时候,需要调用多个方法,然后将各个方法返回的数据一起组装返回给前端,比如这样的:
可以看到我这里调用了4个方法,每一个方法为模拟真实耗时,所以都是延迟100ms返回一个字符串:
可想而知,我们这个接口的响应时间一定会超过400ms,多次执行都会在400ms多一点:
耗时:403ms
耗时:409ms
耗时:406ms
这就是顺序执行,也许大家觉得很Low,但是想想自己的代码很多时候不就是这样子的么?
线程池+Future并行计算
顺序执行确实很慢,所以我们需要并行执行,即同时调用这四个方法,熟悉Java多线程的都知道,每个方法单独开启一个线程异步去执行就好了,等全部执行完了拿到独立线程执行的结果再组装起来就可以了。
但是每次调用都需要创建四个线程,线程的创建和销毁都是需要开销的,所以我们就有了池化技术。
线程池、数据库的连接池等都是采用的池化技术:预先初始生成创建好的线程,等需要调用的时候拿来即用,线程完成工作后回归空闲状态,等待下一次任务的到来,这样就避免了线程频繁的创建、销毁,提高了程序的响应性能。
所以我们在做并行计算的时候一定要充分的利用线程池的相关技术,关于线程池的技术在我的另外一篇文章单独讲到,不了解的同学可以初步了解一下,面试也是必会题之一:
下面我们直接上代码:
多运行几次,看输出响应时间:
耗时: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来开发之前的代码:
这里可以看到实现方式和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中的多线程编程(超详细总结)
- 下一篇: 全局视角看技术-Java多线程演进史
猜你喜欢
- 2025-01-21 Java多线程知识点详解,包含实例,搞懂多线程看这一篇文章就够了
- 2025-01-21 字节Java全能手册火了!多线程/网络/性能调优/框架啥都有
- 2025-01-21 Java多线程中——部分场景使用实现
- 2025-01-21 探究Java多线程(下篇):理论+工具+性能调优+电商场景下的使用
- 2025-01-21 JAVA多线程实现的四种方式
- 2025-01-21 Java面试必考问题:如何保证多线程的有序性
- 2025-01-21 Java获取多线程执行结果方式的归纳与总结
- 2025-01-21 什么!还不懂Java多线程编程?快来看看阿里资深架构师的熬夜总结
- 2025-01-21 项目案例:Java多线程批量拆分List导入数据库
- 2025-01-21 Redis不是号称单线程效率也很高吗,为什么又采用多线程了?
你 发表评论:
欢迎- 07-10动漫人物像|插画 壁纸 头像 签名 素材
- 07-10运动人物|插画 壁纸 头像 签名 素材
- 07-10动漫人物|插画 壁纸 头像 签名 素材
- 07-10神话人物|插画 壁纸 头像 签名 素材
- 07-10日漫人物像|插画 壁纸 头像 签名 素材
- 07-10 日漫人物|插画 壁纸 头像 签名 素材
- 07-10日漫人物风|插画 壁纸 头像 签名 素材
- 07-10日漫人物|插画 壁纸 头像 签名 素材
- 最近发表
- 标签列表
-
- nginx反向代理 (57)
- nginx日志 (56)
- nginx限制ip访问 (62)
- mac安装nginx (55)
- java和mysql (59)
- java中final (62)
- win10安装java (72)
- java启动参数 (64)
- java链表反转 (64)
- 字符串反转java (72)
- java逻辑运算符 (59)
- java 请求url (65)
- java信号量 (57)
- java定义枚举 (59)
- java字符串压缩 (56)
- java中的反射 (59)
- java 三维数组 (55)
- java插入排序 (68)
- java线程的状态 (62)
- java异步调用 (55)
- java中的异常处理 (62)
- java锁机制 (54)
- java静态内部类 (55)
- java怎么添加图片 (60)
- java 权限框架 (55)
本文暂时没有评论,来添加一个吧(●'◡'●)