一、显示地创建线程

Java中显示地创建一个线程,只需要创建Thread(或者其子类,或者实现了Runnable的类)的实例即可。通过调用start()方法就可以启动该线程。

例子:

Thread thread = new Thread(task){
	@Override
	public void run() {
		// Blablabal
	}
};

thread.start();



当在主函数中执行上述代码片段的时候,该任务将被从主线程中分离出来,在它自己的线程中去执行。但是如果我们简简单单的为每一个任务创建一个线程,那么在任务过多的时候将会出现问题:

1. 消耗资源:活跃的线程将消耗系统资源,尤其是内存;

2. 额外的开销:线程的创建,切换与销毁都是需要开销的;

3. 稳定性:线程数是有个上限的,超过它可能会导致OutOfMemoryError,而且这个上限又是与平台相关的。


二、Executor接口

Executor本身是一个接口,它的定义非常简单,如下:

public interface Executor {
	void executor(Runnable command);
}

然而Executor是非常灵活的,因为它把任务的提交与执行解耦开来,从而使系统的线程数可控,而并非由任务数决定。例子:

public class DemoExecutor implements Executor {
	@Override
	public void execute(Runnable command) {
		new Thread(command).start();
	}
}

Executor e = new DemoExecutor();
Runnable task = new Runnable() {
	@Override
	public void run() {
		// Blablabla
	}
};
e.execute(task);

Executor实际上是一种生产者—消费者模式,提交任务相当于生产者,而执行任务是消费者。所以说Executor很适合在生产消费模式的系统里面使用。要注意到的是,把任务的提交和执行分离开来,使我们能够更加轻松地实现一些机制和策略,例如说我们可以在executor中定义是否执行某个线程等等。


三、管理Executor生命周期:ExecutorService

ExecutorService扩展了Executor接口,为其提供了管理生命周期的方法(也有一些便于任务提交的方法):

public interface ExecutorService extends Executor {
    void shutdown();  
    List<Runnable> shutdownNow();  
    boolean isShutdown();  
    boolean isTerminated();  
    boolean awaitTermination(long timeout, TimeUnit unit)  
  
    <T> Future<T> submit(Callable<T> task);  
    <T> Future<T> submit(Runnable task, T result);  
    Future<?> submit(Runnable task);  
    <T> List<Future<T>> invokeAll(Collection<? extends Callable<T>> tasks)  
    <T> List<Future<T>> invokeAll(Collection<? extends Callable<T>> tasks, long timeout, TimeUnit unit)  
    <T> T invokeAny(Collection<? extends Callable<T>> tasks)  
    <T> T invokeAny(Collection<? extends Callable<T>> tasks,  
                    long timeout, TimeUnit unit)  
}

三种状态:关闭,终止,运行。

shutdown是一种平缓的关闭Executor的方式——不接受新的任务,完成已经提交的任务

shutdownNow是比较暴力的,它会试图强制关闭所有执行中的任务,同时当然也不会再接受新任务

关闭后的接受的任务将会由Reject Execution Handler来处理。

awaitTermination阻塞直到所有任务完成,响应中断,可以设置timeout。当awaitTermination之后,通常会shutdown,以关闭Executor。

ExecutorService的例子:

public class DemoExecutorService {
	private final ExecutorService exec = Executors.newSingleThreadExecutor();
	
	private void handler() {
		// Blablabla
	}
	
	public void run() {
		while (!exec.isShutdown()) {
			try {
				exec.execute(new Runnable() {
					@Override
					public void run() {
						handler();
					}
				});
			}catch (RejectedExecutionException e) {
				// Handling exception
			}
		}
	}
}




四、一些线程池

线程池维护了一定数量的可重用的线程。我们可以想象它的好处:1. 线程池的规模是可控的,防止线程数量过多的情况;2. 线程是可复用的,减少了创建销和毁线程的开销。

Executors类的静态工厂方法newXXThreadPool提供了创建和配置以下几种线程池的方法:

1. newFixedThreadPool

固定大小的线程池,每当提交一个任务时就创建一个,直到线程数量达到最大值。如果有异常结束的线程,会有新的线程被补充进来。

2. newCachedThreadPool

有弹性的线程池,没有线程数量的限制,会根据具体环境回收和创建线程。

3. newSingleThreadExecutor

单个线程的线程池,如果异常结束会有新的线程替代。

4. newScheduledThreadPool

固定大小的线程池。Scheduled表明这是一个定时的(或者延迟的)执行任务的线程池。


五、Future & Callable

由于有些运算是很耗时间的,而有的时候我们并不一定要第一时间拿到结果,那么就可以通过Future和Callable这两个接口来实现。

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

public interface Future<V> {
	boolean cancel(boolean mayInterruptIfRunning);
	boolean isCancelled();
	boolean isDone();
	V get() throws InterruptedException, ExecutionException,CancellationException;
	V get(long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, CancellationException;
}

Callable和Runnable很类似,Callable的优点在于 1. 可以有返回值;2. 可以抛出异常。

Future表明任务的生命周期,其核心在于get方法。get会阻塞直到任务完成,或者抛出异常。Future的获取方式很多,其中之一是通过ExecutorService的submit方法提交一个Callable。

例子:

ExecutorService exec = Executors.newSingleThreadExecutor();
Future<void> future = exec.submit(new Callable<void> { public void call() { // Blablabla } });
try {
	future.get();
} catch (InterruptedException e) {
	// Blablabla
} catch (CancellationException e) {
	//Blablabla
} catch (ExecutionException e) {
	//Blablabla
}



另外提一下FutureTask,它实现了Future和Runnable两个接口,因此既可以作为Future又可以被Executor执行。


六、总结

Executor框架把提交与执行解耦,使我们可以更加容易地实现灵活的执行机制和线程管理。