上面是网上见到的一张图,借用一下,
Executor:是Java线程池的超级接口;提供一个execute(Runnable command)方法;我们一般用它的继承接口ExecutorService。
ExecutorService:
ExecutorService也只是个继承接口,主要是三个submit函数,invokeAll是一次调用很多线程。
首先介绍一下submit(Runnable),其实是调用了execute方法的,但是,submit是会返回值的,而且execute是接受Runnable接口的,但是Execute没有返回值。所以,如果submit(Runnable)就需要两次适配。首先,Runnable通过Executors里面的RunnableAdapter适配器,转成有返回值的Callable,然后Callable通过newTaskFor,通过FutureTask方式,转成Runnable。所以submit(Callable)只需要FutureTask即可。(FutureTask实现了Runnable和Future,这相当于一种整合,或者适配,再run里面调用callable,然后返回值,这也是execute没有返回值的结果。)
其实,execute需要一个Runnable,但是有需要返回值,所以就需要适配。很明显,RunnableFuture继承了Runnable,Future,下面是newTaskFor函数的返回结果:
下面转入FutureTask里面,
而FutureTask实现了RunnableFuture, 转入Executors,callable函数实现了将一个Runnable转成Callable类型,需要另一个类,
RunnableAdapter,
可以很清楚的看懂Runnable的适配逻辑。
经过上面的过程,一个Runnable就先转成了一个Callable,再转成Runnable(有返回值)。
这就是submit(Runnable) 逻辑。但是,如果是submit(Callable),就只需要到FutureTask这里就可以了。因为FutureTask就是将Callable适配成Runnable的。
下面说一下Executors,四种线程池方法都在这里,而ThreadPoolExector和ScheduledThreadPoolExector则是真正的线程池。但是和Executors没有继承关系。
Executors:是java.util.concurrent包下的一个类,提供了若干个静态方法,用于生成不同类型的线程池。Executors一共可以创建下面这四类线程池:
- newFixedThreadPool创建一个可缓存线程池,如果线程池长度超过处理需要,可灵活回收空闲线程,若无可回收,则新建线程。
- newFixedThreadPool 创建一个定长线程池,可控制线程最大并发数,超出的线程会在队列中等待。
- newScheduledThreadPool 创建一个线程池,它可安排在给定延迟后运行命令或者定期地执行。
- newSingleThreadExecutor 创建一个使用单个 worker 线程的 Executor,以无界队列方式来运行该线程。它只会用唯一的工作线程来执行任务,保证所有任务按照指定顺序(FIFO, LIFO, 优先级)执行。
补充:
还有创建一个线程的三种方法:
Thread,Runnable,Callable,只有callable是有返回值的。而且,Callable的返回值是Future<?> ,调用过程: 实现Callable,然后new Future(new Callable), new Thread(Future)
- 一个调用Callable的实例,