Java8 - Future 接口

简介: Java8 - Future 接口

20200510181139786.png


Pre

并不希望因为等待某些服务的响应,阻塞应用程序的运行,浪费CPU时钟周期。


这些场景体现了多任务程序设计的另一面。我们前面学习的分支/合并框架以及并行流是实现并行处理的宝贵工具;它们将一个操作分为多个子操作,在多个不同的核、CPU甚至是机器上并行地执行这些子操作。


与此相反,如果你的意图是实现并发,而非并行,或者你的主要目标是在同一个CPU上执行几个松耦合的任务,充分利用CPU的核,让其足够忙碌,从而最大化程序的吞吐量,那么你其实真正想做的是避免因为等待远程服务的返回,或者对数据库的查询,而阻塞线程的执行,浪费宝贵的计算资源,因为这种等待的时间很可能相当长。


Future 接口,尤其是它的新版实现 CompletableFuture ,是处理这种情况的利器 .


并行 VS 并发

20210405215512165.png


Future接口


Future 接口在Java 5中被引入,设计初衷是对将来某个时刻会发生的结果进行建模。它建模了一种异步计算,返回一个执行运算结果的引用,当运算结束后,这个引用被返回给调用方。


在Future 中触发那些潜在的耗时的操作把调用线程解放出来,让它能继续执行其他有价值的工作,不再需要呆呆的等待耗时的操作完成。


打个比方,你可以把它想象成这样的场景:你拿了一袋子衣服到干洗店洗。干洗店的员工会给你张发票,告诉你什么时候你的衣服会洗好(这就是一个 Future 事件) 。 衣服干洗的同时,你可以去做其他的事情。


Future 的另一个优点是它比更底层的 Thread 更易用。要使用 Future ,通常你只需要将耗时的操作封装在一个 Callable 对象中,再将它提交给 ExecutorService ,就OK了。


来看下例子

import java.util.concurrent.*;
/**
 * @author 小工匠
 * @version 1.0
 * @description: TODO
 * @date 2021/4/5 9:47
 * @mark: show me the code , change the world
 */
public class FutureTest {
    public static void main(String[] args) throws ExecutionException, InterruptedException, TimeoutException {
        ExecutorService executorService = Executors.newSingleThreadExecutor();
        Future<String> future = executorService.submit(() -> {
            try {
                TimeUnit.SECONDS.sleep(2);
                return "I'm OK ";
            } catch (InterruptedException e) {
                return "I'm Error ";
            }
        });
        while(!future.isDone()){
            Thread.sleep(10);
        }
        // 超时时间的阻塞
        // future.get(10,TimeUnit.SECONDS);
        // 调用get 阻塞
        System.out.println(future.get());
        executorService.shutdown();
    }
}


使用 Future 以异步的方式执行一个耗时的操作

20210405220144976.png


这种编程方式让你的线程可以在 ExecutorService 以并发方式调用另一个线程执行耗时操作的同时,去执行一些其他的任务。


接着,如果你已经运行到没有异步操作的结果就无法继续任何有意义的工作时,可以调用它的 get 方法去获取操作的结果。


如果操作已经完成,该方法会里立刻返回操作的结果,否则它会阻塞你的线程,直到操作完成,返回相应的结果。


你能想象这种场景存在怎样的问题吗?如果该长时间运行的操作永远远不返回了会怎样?为了处理这种可能性,虽然 Future 提供了一个无需任何参数的 get 方法,我们还是推荐大家使用重载版本的 get 方法,它接受一个超时的参数,通过它,你可以定义你的线程等待 Future 结果的最长时间,而不是一直等待下去。


【使用 Future 以异步方式执行长时间的操作】


20210405220441329.png


Future接口的局限性



通过上面的例子,我们知道 Future 接口提供了方法来检测异步计算是否已经结束(使用isDone 方法),等待异步操作结束 ,以及获取计算的结果。


但是这些特性还不足以让你编写简洁的并发代码。比如,我们很难表述 Future 结果之间的依赖性;从文字描述上这很简单,“当长时间计算任务完成时,请将该计算的结果通知到另一个长时间运行的计算任务,这两个计算任务都完成后,将计算的结果与另一个查询操作结果合并”。


但是,使用 Future 中提供的方法完成这样的操作又是另外一回事。这也是我们需要更具描述能力的特性的原因,比如下面这些。


将两个异步计算合并为一个——这两个异步计算之间相互独立,同时第二个又依赖于第一个的结果

等待 Future 集合中的所有任务都完成。

仅等待 Future 集合中最快结束的任务完成(有可能因为它们试图通过不同的方式计算同一个值),并返回它的结果。

通过编程方式完成一个 Future 任务的执行(即以手工设定异步操作结果的方式)。

应对 Future 的完成事件(即当 Future 的完成事件发生时会收到通知,并能使用 Future计算的结果进行下一步的操作,不只是简单地阻塞等待操作的结果)。

20210405215656404.png


了解新的 CompletableFuture 类(它实现了 Future 接口)如何利用Java 8的新特性以更直观的方式将上述需求都变为可能。


Stream 和 CompletableFuture 的设计都遵循了类似的模式:它们都使用了Lambda表达式以及流水线的思想。


从这个角度,你可以说CompletableFuture 和 Future 的关系就跟 Stream 和 Collection 的关系一样。

相关文章
|
4月前
|
消息中间件 缓存 前端开发
从资损百万到零事故:Java 接口幂等设计的艺术与实践
在分布式系统中,重复请求常引发严重资损,如支付双扣、库存超卖等问题,其根源在于接口缺乏幂等性设计。本文通过真实案例揭示幂等性的重要性,并详解8种主流解决方案,涵盖唯一请求ID、乐观锁、悲观锁、状态机等,帮助开发者构建稳定系统,保障业务一致性。无论你是架构师还是开发工程师,都能从中获得实战指导,有效规避重复调用带来的风险。
296 2
|
4月前
|
数据采集 JSON Java
Java爬虫获取1688店铺所有商品接口数据实战指南
本文介绍如何使用Java爬虫技术高效获取1688店铺商品信息,涵盖环境搭建、API调用、签名生成及数据抓取全流程,并附完整代码示例,助力市场分析与选品决策。
|
3月前
|
算法 安全 Java
除了类,Java中的接口和方法也可以使用泛型吗?
除了类,Java中的接口和方法也可以使用泛型吗?
142 11
|
2月前
|
Java Go 开发工具
【Java】(9)抽象类、接口、内部的运用与作用分析,枚举类型的使用
抽象类必须使用abstract修饰符来修饰,抽象方法也必须使用abstract修饰符来修饰,抽象方法不能有方法体。抽象类不能被实例化,无法使用new关键字来调用抽象类的构造器创建抽象类的实例。抽象类可以包含成员变量、方法(普通方法和抽象方法都可以)、构造器、初始化块、内部类(接 口、枚举)5种成分。抽象类的构造器不能用于创建实例,主要是用于被其子类调用。抽象类中不一定包含抽象方法,但是有抽象方法的类必定是抽象类abstract static不能同时修饰一个方法。
210 0
|
4月前
|
存储 缓存 安全
Java集合框架(二):Set接口与哈希表原理
本文深入解析Java中Set集合的工作原理及其实现机制,涵盖HashSet、LinkedHashSet和TreeSet三大实现类。从Set接口的特性出发,对比List理解去重机制,并详解哈希表原理、hashCode与equals方法的作用。进一步剖析HashSet的底层HashMap实现、LinkedHashSet的双向链表维护顺序特性,以及TreeSet基于红黑树的排序功能。文章还包含性能对比、自定义对象去重、集合运算实战和线程安全方案,帮助读者全面掌握Set的应用与选择策略。
269 23
|
4月前
|
安全 Java 开发者
Java集合框架:详解Deque接口的栈操作方法全集
理解和掌握这些方法对于实现像浏览器后退功能这样的栈操作来说至关重要,它们能够帮助开发者编写既高效又稳定的应用程序。此外,在多线程环境中想保证线程安全,可以考虑使用ConcurrentLinkedDeque,它是Deque的线程安全版本,尽管它并未直接实现栈操作的方法,但是Deque的接口方法可以相对应地使用。
248 12
|
4月前
|
存储 安全 Java
Java集合框架(一):List接口及其实现类剖析
本文深入解析Java中List集合的实现原理,涵盖ArrayList的动态数组机制、LinkedList的链表结构、Vector与Stack的线程安全性及其不推荐使用的原因,对比了不同实现的性能与适用场景,帮助开发者根据实际需求选择合适的List实现。
|
4月前
|
Java API 网络架构
java调用api接口自动判断节假日信息
java调用api接口自动判断节假日信息
1332 0
|
5月前
|
存储 安全 Java
深入理解Java序列化接口及其实现机制
记住,序列化不仅仅是把对象状态保存下来那么简单,它涉及到类的版本控制、安全性和性能等多个重要方面。正确理解和实现Java序列化机制对于构建高效、安全和可维护的Java应用至关重要。
201 0
|
6月前
|
安全 Java API
Java 抽象类与接口在 Java17 + 开发中的现代应用实践解析
《Java抽象类与接口核心技术解析》 摘要:本文全面剖析Java抽象类与接口的核心概念与技术差异。抽象类通过模板设计实现代码复用,支持具体方法与状态管理;接口则定义行为规范,实现多态支持。文章详细对比了两者在实例化、方法实现、继承机制等方面的区别,并提供了模板方法模式(抽象类)和策略模式(接口)的典型应用示例。特别指出Java8+新特性为接口带来的灵活性提升,包括默认方法和静态方法。最后给出最佳实践建议:优先使用接口定义行为规范,通过抽象类实现代码复用,合理组合两者构建灵活架构。
185 2