线程池的原理及实现

转载自   线程池的原理及实现

 

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)来完成了

package mine.util.thread;  import java.util.LinkedList;  
import java.util.List;  /** * 线程池类,线程管理器:创建线程,执行任务,销毁线程,获取线程基本信息 */  
public final class ThreadPool {  // 线程池中默认线程的个数为5  private static int worker_num = 5;  // 工作线程  private WorkThread[] workThrads;  // 未处理的任务  private static volatile int finished_task = 0;  // 任务队列,作为一个缓冲,List线程不安全  private List<Runnable> taskQueue = new LinkedList<Runnable>();  private static ThreadPool threadPool;  // 创建具有默认线程个数的线程池  private ThreadPool() {  this(5);  }  // 创建线程池,worker_num为线程池中工作线程的个数  private ThreadPool(int worker_num) {  ThreadPool.worker_num = worker_num;  workThrads = new WorkThread[worker_num];  for (int i = 0; i < worker_num; i++) {  workThrads[i] = new WorkThread();  workThrads[i].start();// 开启线程池中的线程  }  }  // 单态模式,获得一个默认线程个数的线程池  public static ThreadPool getThreadPool() {  return getThreadPool(ThreadPool.worker_num);  }  // 单态模式,获得一个指定线程个数的线程池,worker_num(>0)为线程池中工作线程的个数  // worker_num<=0创建默认的工作线程个数  public static ThreadPool getThreadPool(int worker_num1) {  if (worker_num1 <= 0)  worker_num1 = ThreadPool.worker_num;  if (threadPool == null)  threadPool = new ThreadPool(worker_num1);  return threadPool;  }  // 执行任务,其实只是把任务加入任务队列,什么时候执行有线程池管理器觉定  public void execute(Runnable task) {  synchronized (taskQueue) {  taskQueue.add(task);  taskQueue.notify();  }  }  // 批量执行任务,其实只是把任务加入任务队列,什么时候执行有线程池管理器觉定  public void execute(Runnable[] task) {  synchronized (taskQueue) {  for (Runnable t : task)  taskQueue.add(t);  taskQueue.notify();  }  }  // 批量执行任务,其实只是把任务加入任务队列,什么时候执行有线程池管理器觉定  public void execute(List<Runnable> task) {  synchronized (taskQueue) {  for (Runnable t : task)  taskQueue.add(t);  taskQueue.notify();  }  }  // 销毁线程池,该方法保证在所有任务都完成的情况下才销毁所有线程,否则等待任务完成才销毁  public void destroy() {  while (!taskQueue.isEmpty()) {// 如果还有任务没执行完成,就先睡会吧  try {  Thread.sleep(10);  } catch (InterruptedException e) {  e.printStackTrace();  }  }  // 工作线程停止工作,且置为null  for (int i = 0; i < worker_num; i++) {  workThrads[i].stopWorker();  workThrads[i] = null;  }  threadPool=null;  taskQueue.clear();// 清空任务队列  }  // 返回工作线程的个数  public int getWorkThreadNumber() {  return worker_num;  }  // 返回已完成任务的个数,这里的已完成是只出了任务队列的任务个数,可能该任务并没有实际执行完成  public int getFinishedTasknumber() {  return finished_task;  }  // 返回任务队列的长度,即还没处理的任务个数  public int getWaitTasknumber() {  return taskQueue.size();  }  // 覆盖toString方法,返回线程池信息:工作线程个数和已完成任务个数  @Override  public String toString() {  return "WorkThread number:" + worker_num + "  finished task number:"  + finished_task + "  wait task number:" + getWaitTasknumber();  }  /** * 内部类,工作线程 */  private class WorkThread extends Thread {  // 该工作线程是否有效,用于结束该工作线程  private boolean isRunning = true;  /* * 关键所在啊,如果任务队列不空,则取出任务执行,若任务队列空,则等待 */  @Override  public void run() {  Runnable r = null;  while (isRunning) {// 注意,若线程无效则自然结束run方法,该线程就没用了  synchronized (taskQueue) {  while (isRunning && taskQueue.isEmpty()) {// 队列为空  try {  taskQueue.wait(20);  } catch (InterruptedException e) {  e.printStackTrace();  }  }  if (!taskQueue.isEmpty())  r = taskQueue.remove(0);// 取出任务  }  if (r != null) {  r.run();// 执行任务  }  finished_task++;  r = null;  }  }  // 停止工作,让该线程自然执行完run方法,自然结束  public void stopWorker() {  isRunning = false;  }  }  
}  

测试代码:

package mine.util.thread;  //测试线程池  
public class TestThreadPool {  public static void main(String[] args) {  // 创建3个线程的线程池  ThreadPool t = ThreadPool.getThreadPool(3);  t.execute(new Runnable[] { new Task(), new Task(), new Task() });  t.execute(new Runnable[] { new Task(), new Task(), new Task() });  System.out.println(t);  t.destroy();// 所有线程都执行完成才destory  System.out.println(t);  }  // 任务类  static class Task implements Runnable {  private static volatile int i = 1;  @Override  public void run() {// 执行任务  System.out.println("任务 " + (i++) + " 完成");  }  }  
}  

 

运行结果:

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提供的线程池更加强大,相信理解线程池的工作原理,看类库中的线程池就不会感到陌生了。

 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/327451.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

计算机考试行高怎么设置,Excel隔行调整行高的四种有效方法

领导要求把一份Excel表格的偶数行行高调整一下。这份表格可是有上百行的&#xff0c;逐一调整行高显然是不科学的。几经周折&#xff0c;费了我N多的口舌四处讨教&#xff0c;这个任务还是顺利地完成了。一、直接定位法先在表格的最后增加一个辅助列。在该列的第一行的单元格中…

亲身体验Intellij Idea从卡顿到顺畅

亲身体验Intellij Idea从卡顿到顺畅 idea power save mode 指尖飘落的程序 2018-01-16 14:12:12 11991 收藏 1 分类专栏&#xff1a; 工具 版权 power save mode 开启后代码不提示&#xff0c; 省电模式&#xff08;经典模式&#xff1a;适用低配版电脑&#xff09; htt…

在收购 Sun 的六年后,Oracle 终于瞄准了 Java 的非付费用户

你还在认为 Java 是免费的吗&#xff1f;再想想&#xff01;&#xff08;2017年你会欠我们很多钱&#xff09; 在收购了 Sun 公司的六年后&#xff0c;Oracle 正在大规模的加大对违反其许可证的 Java 客户的审查力度。 越来越多的 Oracle 客户和合作伙伴被拉里埃里森的团队约谈…

mysql卸载再安装作死级尝试(测试前记得备份数据库)

安装之前一定要将自己的mysql清理干净 1、下载后得到zip压缩包. 2、解压到自己想要安装到的目录&#xff0c;本人解压到的是D:\Environment\mysql-5.7.19 3、添加环境变量&#xff1a;我的电脑->属性->高级->环境变量 选择PATH,在其后面添加: 你的mysql 安装文件下…

按逆向思维定义软件测试,软件测试基础相关概念

测试基础知识是2008-9-20笔记整理&#xff0c;大部份内容来自软件评测师教程&#xff0c;年限已久&#xff0c;有些笔记已丢失。一&#xff0c;理清测试相关概念1&#xff0c;质量 &#xff1a; 实体特性的总和&#xff0c;满足明确或隐含要求的能力2&#xff0c;软件质量 &…

‘entityManagerFactory‘ that could not be found

springboot结合spring data jpa出现a bean named entityManagerFactory that could not be found 羿 2018-10-28 23:46:30 9064 收藏 1 Consider defining a bean named ‘entityManagerFactory’ in your configuration出现的解决方法 原文&#xff1a;https://blog.csdn.…

分享我的第一个asp.net core开发过程

.net core 这个东西感觉还是很不错的&#xff0c;学习了一下&#xff0c;并且做了一个微服务(IP地址查询服务) http://vju.cc/ip/ipquery 看上他的跨平台功能&#xff0c;所以就研究一下&#xff0c;中间有不少坑&#xff0c;有很多第三方类库还不支持.net core&#xff0c;就连…

Java并发编程:深入剖析ThreadLocal

转载自 Java并发编程&#xff1a;深入剖析ThreadLocal 一.对ThreadLocal的理解 ThreadLocal&#xff0c;很多地方叫做线程本地变量&#xff0c;也有些地方叫做线程本地存储&#xff0c;其实意思差不多。可能很多朋友都知道ThreadLocal为变量在每个线程中都创建了一个副本&…

【spring boot】启动类启动 错误: 找不到或无法加载主类 com.codingapi.tm.TxManagerApplication 的解决方案

【spring boot】启动类启动 错误: 找不到或无法加载主类 com.codingapi.tm.TxManagerApplication 的解决方案 导入的一个外部的spring boot项目&#xff0c;运行启动类&#xff0c;出现错误&#xff1a;找不到或无法加载主类 com.codingapi.tm.TxManagerApplication 解决方案…

Rest环境搭建:服务提供者和消费者---SpringCloud

Rest环境搭建&#xff1a;服务提供者和消费者 4.1 介绍 我们会使用一个Dept部门模块做一个微服务通用案例Consumer消费者(Client)通过REST调用Provider提供者(Server)提供的服务。回顾Spring&#xff0c;SpringMVC&#xff0c;Mybatis等以往学习的知识。Maven的分包分模块架构…

Java 多线程 并发编程

转载自 Java 多线程 并发编程 一、多线程 1、操作系统有两个容易混淆的概念&#xff0c;进程和线程。 进程&#xff1a;一个计算机程序的运行实例&#xff0c;包含了需要执行的指令&#xff1b;有自己的独立地址空间&#xff0c;包含程序内容和数据&#xff1b;不同进程的地…

get√—搜索微信公众号【Dotnet跨平台】指定文章的办法

有同学是刚关注“Dotnet跨平台”微信公众号的&#xff0c;结果没看到以前的很多信息&#xff0c;尤其是有用信息&#xff0c;查看历史消息时&#xff0c;有时信息太多&#xff0c;光来回滑动可能要好久&#xff0c;好消息是&#xff0c;微信其实早就支持搜索指定公众号的历史文…

Eureka服务注册中心---SpringCloud

Eureka服务注册中心 5.1 什么是Eureka Netflix在涉及Eureka时&#xff0c;遵循的就是API原则.Eureka是Netflix的有个子模块&#xff0c;也是核心模块之一。Eureka是基于REST的服务&#xff0c;用于定位服务&#xff0c;以实现云端中间件层服务发现和故障转移&#xff0c;服务…

一个复杂系统的拆分改造实践

1 为什么要拆分&#xff1f; 先看一段对话。 从上面对话可以看出拆分的理由&#xff1a; 1&#xff09; 应用间耦合严重。系统内各个应用之间不通&#xff0c;同样一个功能在各个应用中都有实现&#xff0c;后果就是改一处功能&#xff0c;需要同时改系统中的所有应用。这种情…

【深入理解JVM】:类加载器与双亲委派模型

转载自 【深入理解JVM】&#xff1a;类加载器与双亲委派模型 类加载器 加载类的开放性 类加载器&#xff08;ClassLoader&#xff09;是Java语言的一项创新&#xff0c;也是Java流行的一个重要原因。在类加载的第一阶段“加载”过程中&#xff0c;需要通过一个类的全限定名来…

保密计算机能用旧显示器,旧液晶电视机别扔,可作电脑显示器用

现在家里的电视机更新换代也快&#xff0c;可能装修新房就换了一台新液晶电视机。但旧液晶电视机怎么处理&#xff1f;其实&#xff0c;液晶电视机是可以作为台式、笔记本电脑显示器用的。液晶电视也可以看做是电脑的显示屏&#xff0c;但输出亮度较大&#xff0c;长期使用对人…

秒杀架构实践

转载自 秒杀架构实践 前言 本次采用循序渐进的方式逐步提高性能达到并发秒杀的效果&#xff0c;文章较长请准备好瓜子板凳(liushuizhang)。 本文所有涉及的代码&#xff1a; https://github.com/crossoverJie/SSM https://github.com/crossoverJie/distributed-redis-tool 最…

负载均衡Ribbon和Feign---SpringCloud

负载均衡Ribbon和Feign Ribbon负载均衡(基于客户端) 6.1 负载均衡以及Ribbon Ribbon是什么&#xff1f; Spring Cloud Ribbon 是基于Netflix Ribbon 实现的一套客户端负载均衡的工具。简单的说&#xff0c;Ribbon 是 Netflix 发布的开源项目&#xff0c;主要功能是提供客户端…

ASP.NET Core 在 Swagger UI 中显示自定义的 Header Token

Swagger 是个好东西&#xff0c;对于前后端分离的网站来说&#xff0c;不仅是提高前后端开发人员沟通效率的利器&#xff0c;也大大方便了后端人员测试 API。有时候&#xff0c;API 中可能需要在 Header 中设置认证参数&#xff0c;比如 authToken&#xff0c;这样的功能我们通…

nginx,excel模板下载

nginx&#xff0c;excel模板下载 weixin_30814223 2018-08-27 10:26:00 245 收藏 版权 spring boot项目&#xff0c;使用nginx服务器 最近在做一个功能是excel文件上传&#xff0c;并将其中的数据入库&#xff0c;同时还有一个文件模板下载 现在说一说这个文件模板下载 …