Callable和Future

发布时间:2019-11-19 发布网站:脚本宝典
脚本宝典收集整理的这篇文章主要介绍了Callable和Future脚本宝典觉得挺不错的,现在分享给大家,也给大家做个参考。

Callable

相对于Runnable,Callable并不是很让人熟知,其实Callable和Runnable很类似,只不过它有返回值,并且也没有run()方法,而是有call()方法。

public interface Callable<V>{     V call() throw Exception; }

Callable 接口类似于 Runnable,两者都是为那些其实例可能被另一个线程执行的类设计的。但是 Runnable 不会返回结果,并且无法抛出经过检查的异常。

可以看到,返回的结果是以V泛型表示的,比如Callable<Integer>表示一个最终返回Integer的异步计算

@R_406_922@

Future用来保存异步计算的结果,就是说之前用Callable标志的任务可以用Future来进行包装,那为什么非要用Future呢,Callable自己运行然后用相应的类型来接收结果不就行了吗?之所以要用到Future,有一下两个原因

  1. Thread t = new Thread(..)用这个方法创建一个线程,必须要传给一个Runnable的参数,而不能传给它Callable
  2. Future对象具有一系列的方法,比如它的get方法能够被阻塞,直到计算完成;也可以在任务进行的过程中取消它。这些方法使得它更便利

FutureTask

但是Future终究只是一个接口,而FutureTask包装器不仅实现了Future接口,还实现了Runnable接口,这弥补上面的遗憾,使得它不仅能被Thread运行,还具有取消运行的特性,一个典型的使用FutureTask的例子就是:

Callable<Integer> calc = ...; FutureTask<Integer> task = new FutureTask<Integer>(calc); Thread t = new Thread(task);   //这里是Runnable t.start; ... Integer result = task.get();   //这里是Future

住:FutureTask的两个构造方法

FutureTask(Callable<V> task)

FutureTask(Runnable task,V result)

Demo

一个计算指定目录下具有指定关键字的文件数目的例子:

package future;  import java.io.File; import java.io.FileNotFoundException; import java.util.ArrayList; import java.util.List; import java.util.Scanner; import java.util.concurrent.Callable; import java.util.concurrent.Future; import java.util.concurrent.FutureTask;  public class MatchCounter implements Callable<Integer> {  	private File directory; 	private String keyword; 	private int count;  	public MatchCounter(File directory, String keyword) { 		this.directory = directory; 		this.keyword = keyword; 	}  	@Override 	public Integer call() throws Exception { 		count = 0; 		try { 			File[] files = directory.listFiles(); 			List<Future<Integer>> results = new ArrayList<>(); // 用来保存所有的异步计算结果  			for (File file : files) { 				if (file.isDirectory()) { 					MatchCounter counter = new MatchCounter(file, keyword); 					FutureTask<Integer> task = new FutureTask<>(counter); 					results.add(task); 					Thread t = new Thread(task); 					t.start(); 				} else { 					if (search(file)) { 						count++; 					} 				} 			}  			for (Future<Integer> r : results) { 				count += r.get(); 			} 		} catch (InterruptedException e) {  		} 		return count; 	}  	public boolean search(File file) { 		try { 			Scanner in = new Scanner(file); 			boolean found = false; 			while (in.hasNextLine()) { 				String line = in.nextLine(); 				if (line.contains(keyword)) { 					found = true; 				} 			} 			return found; 		} catch (FileNotFoundException e) { 			e.printStackTrace(); 		} 		return false;  	} } 

测试类:

package future;  import java.io.File; import java.util.Scanner; import java.util.concurrent.ArrayBlockingQueue; import java.util.concurrent.BlockingQueue; import java.util.concurrent.ExecutionException; import java.util.concurrent.FutureTask;  public class FutureTest { 	public static void main(String[] args) { 		Scanner in = new Scanner(System.in); 		System.out.println("Enter starting directory:"); 		String d = in.nextLine(); 		System.out.println("Enter keyword:"); 		String keyword = in.nextLine();  		MatchCounter counter = new MatchCounter(new File(d), keyword); 		FutureTask<Integer> task = new FutureTask<>(counter);  		new Thread(task).start();  		try { 			System.out.println(task.get() + " matching files!"); 		} catch (InterruptedException e) { 			e.printStackTrace(); 		} catch (ExecutionException e) { 			e.printStackTrace(); 		} 	} } 

运行:

Enter starting directory: D:workspaceconcurrentsrc Enter keyword: future 2 matching files!

脚本宝典总结

以上是脚本宝典为你收集整理的Callable和Future全部内容,希望文章能够帮你解决Callable和Future所遇到的问题。

如果觉得脚本宝典网站内容还不错,欢迎将脚本宝典推荐好友。

本图文内容来源于网友网络收集整理提供,作为学习参考使用,版权属于原作者。
如您有任何意见或建议可联系处理。小编QQ:384754419,请注明来意。