1、线程池简介:
    多线程技术主要解决处理器单元内多个线程执行的问题,它可以显著减少处理器单元的闲置时间,增加处理器单元的吞吐能力。    
    假设一个服务器完成一项任务所需时间为:T1 创建线程时间,T2 在线程中执行任务的时间,T3 销毁线程时间。

如果:T1 + T3 远大于 T2,则可以采用线程池,以提高服务器性能。
                一个线程池包括以下四个基本组成部分:
                1、线程池管理器(ThreadPool):用于创建并管理线程池,包括 创建线程池,销毁线程池,添加新任务;
                2、工作线程(PoolWorker):线程池中线程,在没有任务时处于等待状态,可以循环的执行任务;
                3、任务接口(Task):每个任务必须实现的接口,以供工作线程调度任务的执行,它主要规定了任务的入口,任务执行完后的收尾工作,任务的执行状态等;
                4、任务队列(taskQueue):用于存放没有处理的任务。提供一种缓冲机制。
                
    线程池技术正是关注如何缩短或调整T1,T3时间的技术,从而提高服务器程序性能的。它把T1,T3分别安排在服务器程序的启动和结束的时间段或者一些空闲的时间段,这样在服务器程序处理客户请求时,不会有T1,T3的开销了。
    线程池不仅调整T1,T3产生的时间段,而且它还显著减少了创建线程的数目,看一个例子:
    假设一个服务器一天要处理50000个请求,并且每个请求需要一个单独的线程完成。在线程池中,线程数一般是固定的,所以产生线程总数不会超过线程池中线程的数目,而如果服务器不利用线程池来处理这些请求则线程总数为50000。一般线程池大小是远小于50000。所以利用线程池的服务器程序不会为了创建50000而在处理请求时浪费时间,从而提高效率。

代码实现中并没有实现任务接口,而是把Runnable对象加入到线程池管理器(ThreadPool),然后剩下的事情就由线程池管理器(ThreadPool)来完成了

[java] view plaincopy
  1. package mine.util.thread;
  2. import java.util.LinkedList;
  3. import java.util.List;
  4. /**
  5. * 线程池类,线程管理器:创建线程,执行任务,销毁线程,获取线程基本信息
  6. */
  7. public final class ThreadPool {
  8. // 线程池中默认线程的个数为5
  9. private static int worker_num = 5;
  10. // 工作线程
  11. private WorkThread[] workThrads;
  12. // 未处理的任务
  13. private static volatile int finished_task = 0;
  14. // 任务队列,作为一个缓冲,List线程不安全
  15. private List<Runnable> taskQueue = new LinkedList<Runnable>();
  16. private static ThreadPool threadPool;
  17. // 创建具有默认线程个数的线程池
  18. private ThreadPool() {
  19. this(5);
  20. }
  21. // 创建线程池,worker_num为线程池中工作线程的个数
  22. private ThreadPool(int worker_num) {
  23. ThreadPool.worker_num = worker_num;
  24. workThrads = new WorkThread[worker_num];
  25. for (int i = 0; i < worker_num; i++) {
  26. workThrads[i] = new WorkThread();
  27. workThrads[i].start();// 开启线程池中的线程
  28. }
  29. }
  30. // 单态模式,获得一个默认线程个数的线程池
  31. public static ThreadPool getThreadPool() {
  32. return getThreadPool(ThreadPool.worker_num);
  33. }
  34. // 单态模式,获得一个指定线程个数的线程池,worker_num(>0)为线程池中工作线程的个数
  35. // worker_num<=0创建默认的工作线程个数
  36. public static ThreadPool getThreadPool(int worker_num1) {
  37. if (worker_num1 <= 0)
  38. worker_num1 = ThreadPool.worker_num;
  39. if (threadPool == null)
  40. threadPool = new ThreadPool(worker_num1);
  41. return threadPool;
  42. }
  43. // 执行任务,其实只是把任务加入任务队列,什么时候执行有线程池管理器觉定
  44. public void execute(Runnable task) {
  45. synchronized (taskQueue) {
  46. taskQueue.add(task);
  47. taskQueue.notify();
  48. }
  49. }
  50. // 批量执行任务,其实只是把任务加入任务队列,什么时候执行有线程池管理器觉定
  51. public void execute(Runnable[] task) {
  52. synchronized (taskQueue) {
  53. for (Runnable t : task)
  54. taskQueue.add(t);
  55. taskQueue.notify();
  56. }
  57. }
  58. // 批量执行任务,其实只是把任务加入任务队列,什么时候执行有线程池管理器觉定
  59. public void execute(List<Runnable> task) {
  60. synchronized (taskQueue) {
  61. for (Runnable t : task)
  62. taskQueue.add(t);
  63. taskQueue.notify();
  64. }
  65. }
  66. // 销毁线程池,该方法保证在所有任务都完成的情况下才销毁所有线程,否则等待任务完成才销毁
  67. public void destroy() {
  68. while (!taskQueue.isEmpty()) {// 如果还有任务没执行完成,就先睡会吧
  69. try {
  70. Thread.sleep(10);
  71. } catch (InterruptedException e) {
  72. e.printStackTrace();
  73. }
  74. }
  75. // 工作线程停止工作,且置为null
  76. for (int i = 0; i < worker_num; i++) {
  77. workThrads[i].stopWorker();
  78. workThrads[i] = null;
  79. }
  80. threadPool=null;
  81. taskQueue.clear();// 清空任务队列
  82. }
  83. // 返回工作线程的个数
  84. public int getWorkThreadNumber() {
  85. return worker_num;
  86. }
  87. // 返回已完成任务的个数,这里的已完成是只出了任务队列的任务个数,可能该任务并没有实际执行完成
  88. public int getFinishedTasknumber() {
  89. return finished_task;
  90. }
  91. // 返回任务队列的长度,即还没处理的任务个数
  92. public int getWaitTasknumber() {
  93. return taskQueue.size();
  94. }
  95. // 覆盖toString方法,返回线程池信息:工作线程个数和已完成任务个数
  96. @Override
  97. public String toString() {
  98. return "WorkThread number:" + worker_num + "  finished task number:"
  99. + finished_task + "  wait task number:" + getWaitTasknumber();
  100. }
  101. /**
  102. * 内部类,工作线程
  103. */
  104. private class WorkThread extends Thread {
  105. // 该工作线程是否有效,用于结束该工作线程
  106. private boolean isRunning = true;
  107. /*
  108. * 关键所在啊,如果任务队列不空,则取出任务执行,若任务队列空,则等待
  109. */
  110. @Override
  111. public void run() {
  112. Runnable r = null;
  113. while (isRunning) {// 注意,若线程无效则自然结束run方法,该线程就没用了
  114. synchronized (taskQueue) {
  115. while (isRunning && taskQueue.isEmpty()) {// 队列为空
  116. try {
  117. taskQueue.wait(20);
  118. } catch (InterruptedException e) {
  119. e.printStackTrace();
  120. }
  121. }
  122. if (!taskQueue.isEmpty())
  123. r = taskQueue.remove(0);// 取出任务
  124. }
  125. if (r != null) {
  126. r.run();// 执行任务
  127. }
  128. finished_task++;
  129. r = null;
  130. }
  131. }
  132. // 停止工作,让该线程自然执行完run方法,自然结束
  133. public void stopWorker() {
  134. isRunning = false;
  135. }
  136. }
  137. }

测试代码:

[java] view plaincopy
  1. package mine.util.thread;
  2. //测试线程池
  3. public class TestThreadPool {
  4. public static void main(String[] args) {
  5. // 创建3个线程的线程池
  6. ThreadPool t = ThreadPool.getThreadPool(3);
  7. t.execute(new Runnable[] { new Task(), new Task(), new Task() });
  8. t.execute(new Runnable[] { new Task(), new Task(), new Task() });
  9. System.out.println(t);
  10. t.destroy();// 所有线程都执行完成才destory
  11. System.out.println(t);
  12. }
  13. // 任务类
  14. static class Task implements Runnable {
  15. private static volatile int i = 1;
  16. @Override
  17. public void run() {// 执行任务
  18. System.out.println("任务 " + (i++) + " 完成");
  19. }
  20. }
  21. }

运行结果:

WorkThread number:3  finished task number:0  wait task number:6
任务 1 完成
任务 2 完成
任务 3 完成
任务 4 完成
任务 5 完成
任务 6 完成
WorkThread number:3  finished task number:6  wait task number:0

分析:由于并没有任务接口,传入的可以是自定义的任何任务,所以线程池并不能准确的判断该任务是否真正的已经完成(真正完成该任务是这个任务的run方法执行完毕),只能知道该任务已经出了任务队列,正在执行或者已经完成。

2、java类库中提供的线程池简介:

     java提供的线程池更加强大,相信理解线程池的工作原理,看类库中的线程池就不会感到陌生了。

其他具体内容查看jdk帮助或看jdk源代码吧。。。

参考文章:http://hi.baidu.com/obullxl/blog/item/ee50ad1ba8e8ff1f8718bf66.html

转载于:https://www.cnblogs.com/kinghitomi/archive/2012/01/19/2327418.html

java多线程总结五:线程池的原理及实现相关推荐

  1. Java多线程系列--“JUC线程池”06之 Callable和Future

    转载自  Java多线程系列--"JUC线程池"06之 Callable和Future Callable 和 Future 简介 Callable 和 Future 是比较有趣的一 ...

  2. 掌握JAVA多线程的利器-线程池

    为什么80%的码农都做不了架构师?>>>    相信大多数接触过多线程的朋友都会有这样的困惑,明明使用了多线程,为何还是一团糟?用下面两幅图再合适不过了: 理想情况下的多线程VS 现 ...

  3. Java多线程-新特性-线程池

    Sun在Java5中,对Java线程的类库做了大量的扩展,其中线程池就是Java5的新特征之一,除了线程池之外,还有很多多线程相关的内容,为多线程的编程带来了极大便利.为了编写高效稳定可靠的多线程程序 ...

  4. Java多线程系列 JUC线程池05 线程池原理解析(四)

    转载 http://www.cnblogs.com/skywang12345/p/3544116.html  https://blog.csdn.net/programmer_at/article/d ...

  5. java多线程编程之线程池技术全面解读

    在多线程编程时,创建线程是十分消耗资源的,当线程创建过多时,便会引发内存溢出,因此引入了线程池技术. 目录 线程池的优势 线程池的创建&使用 线程池的工作原理 线程池的参数 功能线程池 线程池 ...

  6. Java多线程设计模式(4)线程池模式

    前序: Thread-Per-Message Pattern,是一种对于每个命令或请求,都分配一个线程,由这个线程执行工作.它将"委托消息的一端"和"执行消息的一端&qu ...

  7. java多线程抽奖_java 线程池、多线程并发实战(生产者消费者模型 1 vs 10) 附案例源码...

    导读 前二天写了一篇<Java 多线程并发编程>点我直达,放国庆,在家闲着没事,继续写剩下的东西,开干! 线程池 为什么要使用线程池 例如web服务器.数据库服务器.文件服务器或邮件服务器 ...

  8. Java多线程案例之线程池

    文章目录 一. 线程池概述 1. 什么是线程池 2. Java标准库提供的线程池 二. 线程池的简单实现 一. 线程池概述 1. 什么是线程池 线程池和和字符串常量池, 数据库连接池一样, 都是为了提 ...

  9. java多线程并发及线程池

    线程的常用创建方式 1.继承Thread类创建线程类 public class FirstThreadTest extends Thread {public void run(){System.out ...

  10. Java多线程系列 JUC线程池01 线程池框架

    转载  http://www.cnblogs.com/skywang12345/p/3509903.html 为什么引入Executor线程池框架 new Thread()的缺点 1. 每次new T ...

最新文章

  1. easypoi必填项_GitHub - wangjingting/poi-utils: help you more easy read or write excel file
  2. 韶音耳机连不上电脑_骨传导耳机哪个牌子好,骨传导耳机推荐
  3. hdoj-1715-大菲波数(大斐波那契数列)
  4. 晓曼机器人能恢复出厂设置吗_手机出毛病了就恢复出厂设置,会损伤手机吗?影响有多大呢...
  5. 我新买的红米手机,新浪和360浏览器都能进,也能看电视,就是不能上手机QQ和微信...
  6. 若川知乎问答:做前端感觉很吃力怎么办?
  7. 对象中multipartfile 空报错_Python 为什么会有个奇怪的“...”对象?
  8. 作者:李廉(1951-),男,合肥工业大学计算机与信息学院教授
  9. C# 将外部exe程序 嵌入到自己的窗体界面
  10. python移动文件中某个内容_如何在Python中移动文件
  11. PostgreSQL 10.1 手册_部分 II. SQL 语言_第 10 章 类型转换_10.5. UNION、CASE和相关结构...
  12. 如何使用SPSS判断数据的正态分布
  13. ubuntu安装pr_Ubuntu 18.04 安装 Openbravo-3.0PR19Q1
  14. Matlab绘制经纬度地图并添加坐标点
  15. 不良意志品质及其克服
  16. oracle 数据库导入导出文章
  17. 三星910S3L单m2接口更换硬盘系统克隆全程攻略
  18. 小程序接入微信视频号配置指南
  19. paypal IPN and PDT 2
  20. CSS3(新增选择器,nth选择器,伪元素)

热门文章

  1. C语言再学习 -- 输入/输出
  2. 【译】Android中的安全数据-初始化向量
  3. 如何用TensorFlow训练聊天机器人(附github)
  4. Ouroboros:一个可证明安全的PoS区块链协议 (共识介绍)
  5. Android恶意软件偷取Uber凭证
  6. Github 优秀开源项目 Best Open Source Projects
  7. Android开发架构规范
  8. FileProvider 在 Android N 上的应用
  9. android surfaceflinger研究----显示系统
  10. 进击的Android Hook 注入术《二》