java之Thread(3): 线程池

 

java之Thread(3): 线程池_第1张图片

上面是网上见到的一张图,借用一下,

Executor:是Java线程池的超级接口;提供一个execute(Runnable command)方法;我们一般用它的继承接口ExecutorService。

ExecutorService: 

java之Thread(3): 线程池_第2张图片

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没有返回值的结果。)

java之Thread(3): 线程池_第3张图片

java之Thread(3): 线程池_第4张图片

 

其实,execute需要一个Runnable,但是有需要返回值,所以就需要适配。很明显,RunnableFuture继承了Runnable,Future,下面是newTaskFor函数的返回结果:

java之Thread(3): 线程池_第5张图片

下面转入FutureTask里面,

java之Thread(3): 线程池_第6张图片

而FutureTask实现了RunnableFuture, 转入Executors,callable函数实现了将一个Runnable转成Callable类型,需要另一个类,

java之Thread(3): 线程池_第7张图片

RunnableAdapter, 

java之Thread(3): 线程池_第8张图片

可以很清楚的看懂Runnable的适配逻辑。

 

经过上面的过程,一个Runnable就先转成了一个Callable,再转成Runnable(有返回值)。

这就是submit(Runnable) 逻辑。但是,如果是submit(Callable),就只需要到FutureTask这里就可以了。因为FutureTask就是将Callable适配成Runnable的。

 

下面说一下Executors,四种线程池方法都在这里,而ThreadPoolExector和ScheduledThreadPoolExector则是真正的线程池。但是和Executors没有继承关系。

Executors:是java.util.concurrent包下的一个类,提供了若干个静态方法,用于生成不同类型的线程池。Executors一共可以创建下面这四类线程池:

  1. newFixedThreadPool创建一个可缓存线程池,如果线程池长度超过处理需要,可灵活回收空闲线程,若无可回收,则新建线程。

  2. newFixedThreadPool 创建一个定长线程池,可控制线程最大并发数,超出的线程会在队列中等待。

  3. newScheduledThreadPool 创建一个线程池,它可安排在给定延迟后运行命令或者定期地执行。

  4. newSingleThreadExecutor 创建一个使用单个 worker 线程的 Executor,以无界队列方式来运行该线程。它只会用唯一的工作线程来执行任务,保证所有任务按照指定顺序(FIFO, LIFO, 优先级)执行。

java之Thread(3): 线程池_第9张图片

补充:

还有创建一个线程的三种方法:

Thread,Runnable,Callable,只有callable是有返回值的。而且,Callable的返回值是Future ,调用过程: 实现Callable,然后new Future(new Callable), new Thread(Future)

 

  1. 一个调用Callable的实例,

java之Thread(3): 线程池_第10张图片

 

你可能感兴趣的:(java)