前言
最近花了点时间熟悉了下ListenableFuture和CompletableFuture的使用。二者都是原生JDK中老版Future-Get模式的改进。本文将结合demo程序来直观的学习一下这两大Future的使用特点。
老版Future模式的缺点
老版Future模式一个最大的问题是需要获取结果做后续处理操作的时候,还是需要阻塞等待。这样的话,和同步调用方式就没有多大区别了。而ListenableFuture和CompletableFuture对于这种情况则是提供了很多易用的API。
如果说按照先后顺序来讲的话,首先是ListenableFuture,这是由Google Guava工具包提供的Future扩展类,随后,JDK在1.8版本中马上也提供了类似这样的类,就是CompletableFuture。
ListenableFuture
先来聊聊ListenableFuture,一句话概括ListenableFuture和JDK原生Future最大的区别是前者做到了一个可以监听结果的Future。换个更通俗的讲法,就是它可以监听异步执行的过程,执行完了,自动触发什么操作。除此之外,可以分别针对成功的情况,或者失败的情况做各种后续处理。具体使用可以看下面笔者写的demo程序。
package
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.concurrent.Callable;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.Executors;
import org.junit.Test;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.google.common.util.concurrent.FutureCallback;
import com.google.common.util.concurrent.Futures;
import com.google.common.util.concurrent.ListenableFuture;
import com.google.common.util.concurrent.ListeningExecutorService;
import com.google.common.util.concurrent.MoreExecutors;
/**
* The unit test for ListenableFuture/CompletableFuture.
* Created by yiqun01.lin
* on 2018/5/3.
*/
public class TestFutures {
//线程池中线程个数
private static final int POOL_SIZE = 50;
//带有回调机制的线程池
private static final ListeningExecutorService service = MoreExecutors
.listeningDecorator(Executors.newFixedThreadPool(POOL_SIZE));
private static Logger LOG = LoggerFactory.getLogger(TestFutures.class);
@Test
public void testListenableFuture() {
final List<String> value = Collections
.synchronizedList(new ArrayList<String>());
try {
List<ListenableFuture<String>> futures = new ArrayList<ListenableFuture<String>>();
// 将实现了callable的任务放入到线程池中,得到一个带有回调机制的ListenableFuture实例,
// 通过Futures.addCallback方法对得到的ListenableFuture实例进行监听,一旦得到结果就进入到onSuccess方法中,
// 在onSuccess方法中将查询的结果存入到集合中
for (int i = 0; i < 1; i++) {
final