1.直接看代码中的demo:
package com.landon.mavs.example.concurrent;
data:image/s3,"s3://crabby-images/9e1b5/9e1b5b2a3e46b5341b22649797d1794392182f55" alt=""
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
data:image/s3,"s3://crabby-images/9e1b5/9e1b5b2a3e46b5341b22649797d1794392182f55" alt=""
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;
data:image/s3,"s3://crabby-images/9e1b5/9e1b5b2a3e46b5341b22649797d1794392182f55" alt=""
data:image/s3,"s3://crabby-images/2a1f3/2a1f35146451967292b66fa62c8f22027e7067cf" alt=""
/** *//**
*
* 使用guava实现异步回调 {@link java.util.concurrent.Future}
* {@link com.google.common.util.concurrent.ListenableFuture}
* {@link com.google.common.util.concurrent.FutureCallback}
*
* @author landon
*
*/
data:image/s3,"s3://crabby-images/2a1f3/2a1f35146451967292b66fa62c8f22027e7067cf" alt=""
public class FutureCallbackExample
{
data:image/s3,"s3://crabby-images/8d7d9/8d7d99ac571b1efcbf1f7e7a4120707c8e90d1fd" alt=""
public static void main(String[] args) throws Exception
{
// 原生的Future模式,实现异步
ExecutorService nativeExecutor = Executors.newSingleThreadExecutor();
Future<String> nativeFuture = nativeExecutor
data:image/s3,"s3://crabby-images/8d7d9/8d7d99ac571b1efcbf1f7e7a4120707c8e90d1fd" alt=""
.submit(new Callable<String>()
{
data:image/s3,"s3://crabby-images/96c01/96c01a9005d00151a1af2189b6a9f266915ba654" alt=""
@Override
data:image/s3,"s3://crabby-images/8d7d9/8d7d99ac571b1efcbf1f7e7a4120707c8e90d1fd" alt=""
public String call() throws Exception
{
// 使用sleep模拟调用耗时
TimeUnit.SECONDS.sleep(1);
return "This is native future call.not support async callback";
}
});
data:image/s3,"s3://crabby-images/96c01/96c01a9005d00151a1af2189b6a9f266915ba654" alt=""
// Future只实现了异步,而没有实现回调.所以此时主线程get结果时阻塞.或者可以轮训以便获取异步调用是否完成
System.out.println(nativeFuture.get());
data:image/s3,"s3://crabby-images/96c01/96c01a9005d00151a1af2189b6a9f266915ba654" alt=""
// 好的实现应该是提供回调,即异步调用完成后,可以直接回调.本例采用guava提供的异步回调接口,方便很多.
ListeningExecutorService guavaExecutor = MoreExecutors
.listeningDecorator(Executors.newSingleThreadExecutor());
final ListenableFuture<String> listenableFuture = guavaExecutor
data:image/s3,"s3://crabby-images/8d7d9/8d7d99ac571b1efcbf1f7e7a4120707c8e90d1fd" alt=""
.submit(new Callable<String>()
{
data:image/s3,"s3://crabby-images/96c01/96c01a9005d00151a1af2189b6a9f266915ba654" alt=""
@Override
data:image/s3,"s3://crabby-images/8d7d9/8d7d99ac571b1efcbf1f7e7a4120707c8e90d1fd" alt=""
public String call() throws Exception
{
Thread.sleep(1000);
return "this is guava future call.support async callback";
}
});
data:image/s3,"s3://crabby-images/96c01/96c01a9005d00151a1af2189b6a9f266915ba654" alt=""
// 注册监听器,即异步调用完成时会在指定的线程池中执行注册的监听器
data:image/s3,"s3://crabby-images/8d7d9/8d7d99ac571b1efcbf1f7e7a4120707c8e90d1fd" alt=""
listenableFuture.addListener(new Runnable()
{
data:image/s3,"s3://crabby-images/96c01/96c01a9005d00151a1af2189b6a9f266915ba654" alt=""
@Override
data:image/s3,"s3://crabby-images/8d7d9/8d7d99ac571b1efcbf1f7e7a4120707c8e90d1fd" alt=""
public void run()
{
data:image/s3,"s3://crabby-images/8d7d9/8d7d99ac571b1efcbf1f7e7a4120707c8e90d1fd" alt=""
try
{
System.out.println("async complete.result:"
+ listenableFuture.get());
data:image/s3,"s3://crabby-images/8d7d9/8d7d99ac571b1efcbf1f7e7a4120707c8e90d1fd" alt=""
} catch (Exception e)
{
}
}
}, Executors.newSingleThreadExecutor());
data:image/s3,"s3://crabby-images/96c01/96c01a9005d00151a1af2189b6a9f266915ba654" alt=""
// 主线程可以继续执行,异步完成后会执行注册的监听器任务.
System.out.println("go on execute.asyn complete will callback");
data:image/s3,"s3://crabby-images/96c01/96c01a9005d00151a1af2189b6a9f266915ba654" alt=""
// 除了ListenableFuture,guava还提供了FutureCallback接口,相对来说更加方便一些.
ListeningExecutorService guavaExecutor2 = MoreExecutors
.listeningDecorator(Executors.newSingleThreadExecutor());
final ListenableFuture<String> listenableFuture2 = guavaExecutor2
data:image/s3,"s3://crabby-images/8d7d9/8d7d99ac571b1efcbf1f7e7a4120707c8e90d1fd" alt=""
.submit(new Callable<String>()
{
data:image/s3,"s3://crabby-images/96c01/96c01a9005d00151a1af2189b6a9f266915ba654" alt=""
@Override
data:image/s3,"s3://crabby-images/8d7d9/8d7d99ac571b1efcbf1f7e7a4120707c8e90d1fd" alt=""
public String call() throws Exception
{
Thread.sleep(1000);
System.out.println("asyncThreadName:"
+ Thread.currentThread().getName());
return "this is guava future call.support async callback using FutureCallback";
}
});
// 注意这里没用指定执行回调的线程池,从输出可以看出,默认是和执行异步操作的线程是同一个.
data:image/s3,"s3://crabby-images/8d7d9/8d7d99ac571b1efcbf1f7e7a4120707c8e90d1fd" alt=""
Futures.addCallback(listenableFuture2, new FutureCallback<String>()
{
data:image/s3,"s3://crabby-images/96c01/96c01a9005d00151a1af2189b6a9f266915ba654" alt=""
@Override
data:image/s3,"s3://crabby-images/8d7d9/8d7d99ac571b1efcbf1f7e7a4120707c8e90d1fd" alt=""
public void onSuccess(String result)
{
System.out
.println("async callback(using FutureCallback) result:"
+ result);
System.out.println("execute callback threadName:"
+ Thre }ad.currentThread().getName());
data:image/s3,"s3://crabby-images/5b1d8/5b1d83c5952f03a9561a63d240fed8c3222ba678" alt=""
data:image/s3,"s3://crabby-images/96c01/96c01a9005d00151a1af2189b6a9f266915ba654" alt=""
@Override
data:image/s3,"s3://crabby-images/8d7d9/8d7d99ac571b1efcbf1f7e7a4120707c8e90d1fd" alt=""
public void onFailure(Throwable t)
{
}
});
}
}
data:image/s3,"s3://crabby-images/9e1b5/9e1b5b2a3e46b5341b22649797d1794392182f55" alt=""
2.总结:利用guava的ListenableFuture和FutureCallback方便的实现异步回调.
posted on 2014-02-27 14:54
landon 阅读(7184)
评论(3) 编辑 收藏 所属分类:
Program