Java使用线程池的优势有哪些
池化技术相比大家已经屡见不鲜了,线程池、数据库连接池、http 连接池等等都是对这个思想的应用。池化技术的思想主要是为了减少每次获取资源的消耗,提高对资源的利用率。 线程池提供了一种限制和管理资源(包括执行一个任务)。 每个线程池还维护一些基本统计信息,例如已完成任务的数量。
这里借用《java 并发编程的艺术》提到的来说一下使用线程池的好处:
- 降低资源消耗。通过重复利用已创建的线程降低线程创建和销毁造成的消耗。
- 提高响应速度。当任务到达时,任务可以不需要的等到线程创建就能立即执行。
- 提高线程的可管理性。线程是稀缺资源,如果无限制的创建,不仅会消耗系统资源,还会降低系统的稳定性,使用线程池可以进行统一的分配,调优和监控。
executor 框架
executor 框架是 java5 之后引进的,在 java 5 之后,通过 executor 来启动线程比使用 thread 的 start 方法更好,除了更易管理,效率更好(用线程池实现,节约开销)外,还有关键的一点:有助于避免 this 逃逸问题。
补充:this 逃逸是指在构造函数返回之前其他线程就持有该对象的引用. 调用尚未构造完全的对象的方法可能引发令人疑惑的错误。
executor 框架不仅包括了线程池的管理,还提供了线程工厂、队列以及拒绝策略等,executor 框架让并发编程变得更加简单。
executor 框架结构(主要由三大部分组成)
- 任务(runnable /callable) 执行任务需要实现的 runnable 接口 或 callable接口。runnable 接口或 callable 接口 实现类都可以被 threadpoolexecutor 或 scheduledthreadpoolexecutor 执行。
- 任务的执行(executor) 如下图所示,包括任务执行机制的核心接口 executor ,以及继承自 executor 接口的 executorservice 接口。threadpoolexecutor 和 scheduledthreadpoolexecutor 这两个关键类实现了 executorservice 接口。
这里提了很多底层的类关系,但是,实际上我们需要更多关注的是 threadpoolexecutor 这个类,这个类在我们实际使用线程池的过程中,使用频率还是非常高的。
注意: 通过查看 scheduledthreadpoolexecutor 源代码我们发现 scheduledthreadpoolexecutor 实际上是继承了 threadpoolexecutor 并实现了 scheduledexecutorservice ,而 scheduledexecutorservice 又实现了 executorservice,正如我们下面给出的类关系图显示的一样。
threadpoolexecutor 类描述:
//abstractexecutorservice实现了executorservice接口 public class threadpoolexecutor extends abstractexecutorservice
scheduledthreadpoolexecutor 类描述:
//scheduledexecutorservice实现了executorservice接口 public class scheduledthreadpoolexecutor extends threadpoolexecutor implements scheduledexecutorservice
3) 异步计算的结果(future) future 接口以及 future 接口的实现类 futuretask 类都可以代表异步计算的结果。
当我们把 runnable接口 或 callable 接口 的实现类提交给 threadpoolexecutor 或 scheduledthreadpoolexecutor 执行。(调用 submit() 方法时会返回一个 futuretask 对象)
executor 框架的使用示意图
- 主线程首先要创建实现 runnable 或者 callable 接口的任务对象。
- 把创建完成的实现 runnable/callable接口的 对象直接交给 executorservice 执行: executorservice.execute(runnable command))或者也可以把 runnable 对象或callable 对象提交给 executorservice 执行(executorservice.submit(runnable task)或 executorservice.submit(callable task))。
- 如果执行 executorservice.submit(…),executorservice 将返回一个实现future接口的对象(我们刚刚也提到过了执行 execute()方法和 submit()方法的区别,submit()会返回一个 futuretask 对象)。由于 futuretask 实现了 runnable,我们也可以创建 futuretask,然后直接交给 executorservice 执行。
- 最后,主线程可以执行 futuretask.get()方法来等待任务执行完成。主线程也可以执行 futuretask.cancel(boolean mayinterruptifrunning)来取消此任务的执行。
以上就是java使用线程池的优势有哪些的详细内容,更多关于java 线程池的资料请关注其它相关文章!