10063---java多线程总结五:线程池的原理及实现

news/2024/7/4 1:22:40

原文

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 com.cbg.util.thread;

import java.util.LinkedList;
import java.util.List;

/**
 * 线程池类,线程池管理器:创建线程,执行任务,销毁线程,获取线程基本信息
 */
public class ThreadPool {
	// 线程池中默认线程的个数为5
	private static int worker_num = 5;
	// 工作线程
	private WorkThread[] workThreads;
	// 未处理的任务
	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;
		workThreads = new WorkThread[worker_num];
		for (int i = 0; i < worker_num; i++) {
			workThreads[i] = new WorkThread();
			workThreads[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++) {
			workThreads[i].stopWorker();
			workThreads[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;
		}
	}
}
测试代码:

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



其他具体内容查看jdk帮助或看jdk源代码吧。。。
参考文章:http://hi.baidu.com/obullxl/blog/item/ee50ad1ba8e8ff1f8718bf66.html


http://www.niftyadmin.cn/n/2868287.html

相关文章

卡方检验的自由度

为什么80%的码农都做不了架构师&#xff1f;>>> 自由度&#xff1a;取值不受限制的变量的个数。 如何理解这句简单的话呢&#xff1f;给定一组数据&#xff0c;我们来计算不同的统计量&#xff0c;看看自由度的变化。这些数据分别为 1 2 4 6 8. 5个数。 先来求平均…

大数据里挖掘大商机

“身处阿里巴巴总部的数据直播中心&#xff0c;仿佛来到航天基地观摩火箭发射。”一位参加过现场报道的媒体同行曾向记者这样描述。 作为最受消费者关注的年度盛事之一&#xff0c;11月10日至11日&#xff0c;来自海内外的460多家媒体守候在阿里巴巴西溪园区&#xff0c;捕捉着…

10065---【Java多线程】线程池的工作原理详解(上)

原文 为什么使用线程池 线程池为线程生命周期开销问题和资源不足问题提供了解决方案&#xff0c;因为线程是稀缺资源&#xff0c;如果被无限制的创建&#xff0c;不仅会消耗系统资源&#xff0c;还会降低系统的稳定性&#xff0c;合理的使用线程池对线程进行统一分配、调优和监…

使用vue-meta进行mate动态管理HTML head信息

vue-meta的官方文档在这里。文档中比较详细地说明了在浏览器端和服务器端如何使用 vue-meta 修改页面头部信息&#xff0c;这里我主要介绍下在SPA项目中管理meta info的使用方法。&#xff0c; vue单页运用中&#xff0c;对单独页面使用meta的时候&#xff0c;他不是直接修改&a…

大数据会带来越来越细的标准

我们经常谈大数据&#xff0c;什么是大数据呢&#xff1f;大数据首先它体积非常大&#xff0c;有各种各样的数据&#xff0c;另外还有非常复杂&#xff0c;还包括数据传输的速度。在2000年的时候&#xff0c;我们的因特网的大小规模大概21Tb&#xff0c;用现在的技术&#xff0…

10066--- java线程:互斥锁与读写锁

原文 两种互斥锁机制&#xff1a; 1、synchronized 2、ReentrantLock ReentrantLock是jdk5的新特性&#xff0c;采用ReentrantLock可以完全替代替换synchronized传统的锁机制&#xff0c;而且采用ReentrantLock的方式更加面向对象&#xff0c;也更加灵活&#xff0c;网上有很多…

你不知道的天猫“双十一”:大数据成为新驱动力

11月11日&#xff0c;时间刚过中午12点&#xff0c;杭州阿里巴巴西溪园区的天猫双十一数据直播大屏上显示的交易额数据已经超过327亿元。仅用了半天&#xff0c;成交即接近2013年“双十一”全天的数字。 全球最高单日网购纪录再次被打破。不过&#xff0c;在阿里巴巴的数据专…

10025---sort +awk+uniq 统计文件中出现次数最多的前10个单词

原文 使用linux命令或者shell实现&#xff1a;文件words存放英文单词&#xff0c;格式为每行一个英文单词&#xff08;单词可以重复&#xff09;&#xff0c;统计这个文件中出现次数最多的前10个单词。 cat words.txt | sort | uniq -c | sort -k1,1nr | head -10主要考察对s…