java多线程学习(2)

1)Callable和Future

  

Runnable封装一个异步运行的任务;可以当成一个没有任何参数和返回值的异步方法,Callable和 Runnable类似,但是它有返回值和参数。

  

Callable接口是一个参数化的类型,只有一个方法call。

1 public interface Callable<V>
2 
3 {
4   
5    V call()throws Exception;
6 
7 }

类型参数v是指返回值的类型,例如Callable<Integer>代表最终返回一个Integer类型对象的异步计算。

 

Future保存异步计算的结果。Future接口具有下面的方法:

 1 public interface Future<V>
 2 {
 3   
 4    V get() throws....;
 5 
 6    V get(long timeout, TimeUnit unit) throws...;
 7 
 8    void cancel(boolean mayInterrupt);
 9 
10    boolean isCancelled();
11 
12    boolean isDone();
13 
14 
15 }

第一个get方法的调用将被阻塞,直到计算完成,第二个get方法调用,如果计算完成之前超时,将抛出异常;如果计算还在进行,isdone方法返回false;如果想取消计算,调用cancel方法,如果计算还没开始,将永远也不会开始,如果计算已经开始,那么,如果参数mayInterrupt设为true,则被打断。

 

FutureTask包装器是一种很方便的将Callable转换成Future和Runnable的机制,它同时实现了这两个接口。例如:

 1 Callable<Integer> myComputation=....;
 2 
 3 FutureTask<Integer> task=new FutureTask<Integer>(myComputation);
 4 
 5 Thread t=new Thread(task);//it's a Runnable;
 6 
 7 t.start();
 8 
 9 ...
10 
11 Integer result=task.get();//it's a Future;

 

2)线程池

构建一个新的线程还是需要一些开销的,如果需要创建大量的生存期很短的线程,那就应该使用线程池,一个线程池包含大量的空闲线程;将一个Runnable对象给线程池,线程池中的一个线程就会调用run方法,当run方法结束之后,该线程并不会死亡,而是继续在池中准备为下一个请求提供服务。

执行器(Executor)类有大量用来构建线程池的静态工厂方法,常用的有以下三个方法:

newCachedThreadPool方法,构建一个线程池,对于任务请求,如果有空闲线程可用,立即让它执行任务,否则就创建一个新线程。

newFixedThreadPool方法,构建一个固定大小的线程池,如果提交的任务大于空闲线程的数量,那么得不到服务的任务将被置于等待队列中。

newSingleThreadExecutor,构建一个大小为1的线程池,由一个线程执行所有任务,一个接一个。

使用方法,例如:

 1 ExecutorService pool=Executors.newCachedThreadPool();
 2 
 3  Callable<Integer> myComputation=....;
 4  
 5  Future<Integer> task=pool.submit(myComputation);
 6 
 7   ...
 8  
 9  Integer result=task.get();
10 
11 pool.shutdown();

 

 

2014-07-22 20:53:08

posted @ 2014-07-22 20:54  烈焰中的清凉  阅读(160)  评论(0编辑  收藏  举报