线程池介绍:


多线程技术主要解决处理器单元内多个线程执行的问题,它可以显著减少处理器单元的闲置时间,


增加处理器单元的吞吐能力。 假设一个服务器完成一项任务所需时间为:T1 创建线程时间, T2 在


线程中执行任务的时间,T3 销毁线程时间。



涉及的接口与类:

一.Java中的ThreadPoolExecutor类(该类有四种构造方法)

public ThreadPoolExecutor(int corePoolSize,int maximumPoolSize,long keepAliveTime,TimeUnit unit, 
  
 
BlockingQueue<Runnable> workQueue);




     public ThreadPoolExecutor(int corePoolSize,int maximumPoolSize,long keepAliveTime,TimeUnit unit,
 
 
 
   
  
BlockingQueue<Runnable> workQueue,ThreadFactory threadFactory);





   public ThreadPoolExecutor(int corePoolSize,int maximumPoolSize,long keepAliveTime,TimeUnit unit,
 
  
BlockingQueue<Runnable> workQueue,RejectedExecutionHandler handler);




   public ThreadPoolExecutor(int corePoolSize,int maximumPoolSize,long keepAliveTime,TimeUnit unit,
 
  
BlockingQueue<Runnable> workQueue,ThreadFactory threadFactory,RejectedExecutionHandler handler);








二.Java中的ExecutorService接口



可缓存线程池



Executors.newCachedThreadPool();





定长线程池



Executors.newFixedThreadPool(3);





定长线程池,定时器




  Executors.newScheduledThreadPool(3);





单线程化的线程池



Executors.newSingleThreadExecutor();



三.Java中的ScheduledExecutorService接口


代码实现:代码中已经对各个方法有详细的注释,请仔细阅读注释;






package com.superb.juint_thread;

import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;

/**
 * 该类中有五种实现线程池的方法,其中有三种通过ExecutorServic接口实现,一种
 * 通过ScheduledExecutorService接口实现,另一种通过ThreadPoolExecutor类实现
 * 
 * 这里面将五种线程池用单例模式实现,当然也没必要用单例模式实现,这个完全根据项目的业务需要来定
 * 
 * @author superb
 *
 */
public class ThreadPoolUtil {
	
	private static ThreadPoolExecutor pool = null;

	/** 可缓存线程池 */
	private static ExecutorService cachedPool = null;
	
	/** 定长线程池 */
	private static ExecutorService fixedPool = null;
	
	/** 定长线程池,支持定时及周期性 */
	private static ScheduledExecutorService schedulPool = null;
	
	/** 单例线程池 */
	private static ExecutorService singlePool = null;
	
	
	private ThreadPoolUtil() {
	}
	
	/**
	 * 得到一个ThreadPoolExecutor线程池
	 * new ThreadPoolExecutor(int corePoolSize, int maximumPoolSize, 
	 * 			long keepAliveTime, TimeUnit unit, BlockingQueue
    
    
     
      workQueue, 
	 * 			RejectedExecutionHandler handler)
	 * 
	 * @corePoolSize 线程池大小
	 * @maximumPoolSize 线程池最大值
	 * @keepAliveTime 线程保存的存活时间
	 * @workQueue 线程数超过线程大小时,任务存在这个队列中
	 * @handler 线程超过队列时,处理办法
	 * @return
	 */
	public static ThreadPoolExecutor newInstance(){
		if(pool==null){
			synchronized(ThreadPoolUtil.class){
				if(pool==null){
					pool = new ThreadPoolExecutor(3, 5, 10, TimeUnit.SECONDS,
							new ArrayBlockingQueue
     
     
      
      (10), new ThreadPoolExecutor.DiscardOldestPolicy());
				}
			}
		}
		return pool;
	}
	
	/**
	 * 创建一个可缓存线程池,如果线程池长度超过处理需要,可灵活回收空闲线程,若无可回收,则新建线程。
	 * 
	 * 线程池为无限大,当执行第二个任务时第一个任务已经完成,会复用执行第一个任务的线程,而不用每次新建线程。 
	 * @return CachedThreadPool
	 */
	public static ExecutorService cachedInstance(){
		if(cachedPool == null){
			synchronized (ThreadPoolUtil.class) {
				if(cachedPool==null){
					cachedPool = Executors.newCachedThreadPool();
				}
			}
		}	
		return cachedPool;
	} 
	
	/**
	 * 创建一个定长线程池,可控制线程最大并发数,超出的线程会在队列中等待。(这里线程池大小为3)
	 * 
	 * 定长线程池的大小最好根据系统资源进行设置
	 * 
	 * @return CachedThreadPool
	 */
	public static ExecutorService fixedInstance(){
		if(fixedPool == null){
			synchronized (ThreadPoolUtil.class) {
				if(fixedPool==null){
					fixedPool = Executors.newFixedThreadPool(3);
				}
			}
		}	
		return fixedPool;
	} 
	
	/**
	 * 创建一个定长线程池,支持定时及周期性任务执行,(定时任务)
	 * 
	 * @return CachedThreadPool
	 */
	public static ScheduledExecutorService schedulInstance(){
		if(schedulPool == null){
			synchronized (ThreadPoolUtil.class) {
				if(schedulPool==null){
					schedulPool = Executors.newScheduledThreadPool(3);
				}
			}
		}	
		return schedulPool;
	} 
	
	/**
	 * 创建一个单线程化的线程池,它只会用唯一的工作线程来执行任务,保证所有任务按照指定顺序(FIFO, LIFO, 优先级)执行
	 * 
	 * 顺序执行各个任务。
	 * @return CachedThreadPool
	 */
	public static ExecutorService singleInstance(){
		if(singlePool == null){
			synchronized (ThreadPoolUtil.class) {
				if(singlePool==null){
					singlePool = Executors.newScheduledThreadPool(10);
				}
			}
		}	
		return singlePool;
	} 
}
package com.superb.juint_thread;

import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;

/**
 * 测试线程池的实现main类
 *
 */
public class ThreadMain {
    public static void main( String[] args ){
    	//singleMethod();
    	//cachedMethod();
    	//schedulMethod();
    	//fixedMethod();
    	threadPoolMethod();
    }
    
    /**
     * 描述:
     * 		得到一个ThreadPoolExecutor线程池,
     * 		通过new ThreadPoolExecutor()对象类来实现。
     * 		该方法new一个 (@corePoolSize) 线程池大小为 3 的线程池。
     * 		模拟任务有10个,每次执行三个,剩余的放在队列中等待执行
     * 
     * result:
     * ---------------BEGIN--------------
     * ---------------END--------------
     * -----------pool-1-thread-1---2016-08-12 05:47:13
     * -----------pool-1-thread-2---2016-08-12 05:47:13
     * -----------pool-1-thread-3---2016-08-12 05:47:13
     * 
     * -----------pool-1-thread-1---2016-08-12 05:47:15
     * -----------pool-1-thread-3---2016-08-12 05:47:15
     * -----------pool-1-thread-2---2016-08-12 05:47:15
     * 
     * -----------pool-1-thread-1---2016-08-12 05:47:17
     * -----------pool-1-thread-3---2016-08-12 05:47:17
     * -----------pool-1-thread-2---2016-08-12 05:47:17
     * 
     * -----------pool-1-thread-3---2016-08-12 05:47:19
     */
    public static void threadPoolMethod(){
    	ThreadPoolExecutor pool = ThreadPoolUtil.newInstance();
		System.out.println("---------------BEGIN--------------");
    	for(int i = 0; i < 10; i++){
    		pool.execute(new Runnable() {
    			public void run() {
    				System.out.println("-----------"+Thread.currentThread().getName()+
    						"---"+new SimpleDateFormat("yyyy-MM-dd hh:mm:ss").format(new Date()));
    				try {
						Thread.sleep(2000);
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
    			}
    		});
    	}
		System.out.println("---------------END--------------");
    }
    
    /**
     * 描述:
     * 		定时器,可以指定业务每隔指定时间执行一次,该方法中每隔 3 秒模拟一次业务处理
     * result:
     * 
     * 模拟业务处理=====2016-08-12 04:30:37
	 * 模拟业务处理=====2016-08-12 04:30:40
	 * 模拟业务处理=====2016-08-12 04:30:43
	 * 模拟业务处理=====2016-08-12 04:30:46
	 * 模拟业务处理=====2016-08-12 04:30:49
     */
    public static void schedulMethod(){
    	ScheduledExecutorService pool = ThreadPoolUtil.schedulInstance();
    	pool.scheduleAtFixedRate(new Runnable() {
			public void run() {
				System.out.println("模拟业务处理====="+new SimpleDateFormat("yyyy-MM-dd hh:mm:ss")
						.format(new Date()));
			}
		}, 0, 3, TimeUnit.SECONDS);
    }
    
    /**
     * 描述:
     * 		线程池为无限大,当执行第二个任务时第一个任务已经完成,会复用执行第一个任务的线程,而不用每次新建线程。
	 * 结论:
	 * 		当每个任务执行5s时,任务间隔为2s时,上一个任务未执行完时,下一个任务已经开始,此时不会复用上一个线程。
	 * 		当任务执行为2s,任务间隔为5s时,下一个任务开始时,上一个任务已完成,此时会复用上一个线程
	 *  result 1:
	 *  
     
     
	 *	
     
     ------>pool-1-thread-1<------
	 *	
     
     ------>pool-1-thread-2<------
	 *	
     
     ------>pool-1-thread-3<------
	 *	
     
     ------>pool-1-thread-1<------
	 *	
     
     ------>pool-1-thread-2<------
	 *	
     
     ------>pool-1-thread-3<------
	 *	
     
     ----------->SUCCESS<------
	 *
	 * 	result 2:
	 *  
     
     ----------->BEGIN<-------
	 *	
     
     ------>pool-1-thread-1<------
	 *	
     
     ------>pool-1-thread-1<------
	 *	
     
     ------>pool-1-thread-1<------
	 *	
     
     ------>pool-1-thread-1<------
	 *	
     
     ----------->SUCCESS<------
	 */
	public static void cachedMethod() {
    	ExecutorService pool = ThreadPoolUtil.cachedInstance();
		System.out.println("
     
     ----------->BEGIN<-------");
		for (int i = 0; i < 10; i++) {
			pool.execute(new Runnable() {
				public void run() {
					System.out.println("
     
     ------>" + Thread.currentThread().getName() + "<------");
					try {
						Thread.sleep(5000);// 每个任务执行5s,
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
				}
			});
			try {
				Thread.sleep(2000);// 每个任务间隔2s
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}

		System.out.println("
     
     ----------->SUCCESS<------");
    }

	/**
	 * 描述:
	 * 		定长线程池,可控制线程最大并发数,超出的线程会在队列中等待,
	 * 		该方法中我们实行了一个长度为3的线程池,其中开启10个任务,每次只执行3个任务,
	 * 		剩余任务会在队列中等待,直到有线程空闲执行下一个任务
	 * result:
	 * ---------------BEGIN--------------
	 * ---------------END--------------
	 * ----模拟任务----pool-1-thread-2----2016-08-12 05:09:02
	 * ----模拟任务----pool-1-thread-1----2016-08-12 05:09:02
	 * ----模拟任务----pool-1-thread-3----2016-08-12 05:09:02
	 * 
	 * ----模拟任务----pool-1-thread-3----2016-08-12 05:09:07
	 * ----模拟任务----pool-1-thread-2----2016-08-12 05:09:07
	 * ----模拟任务----pool-1-thread-1----2016-08-12 05:09:07
	 * 
	 * ----模拟任务----pool-1-thread-3----2016-08-12 05:09:12
	 * ----模拟任务----pool-1-thread-2----2016-08-12 05:09:12
	 * ----模拟任务----pool-1-thread-1----2016-08-12 05:09:12
	 * 
	 * ----模拟任务----pool-1-thread-3----2016-08-12 05:09:17
	 * 
	 */
	public static void fixedMethod() {
		ExecutorService pool = ThreadPoolUtil.fixedInstance();
		System.out.println("---------------BEGIN--------------");
		for (int i = 0; i < 10; i++) {
			pool.execute(new Runnable() {
				
				public void run() {
					System.out.println("----模拟任务----"+Thread.currentThread().getName()
							+ "----"+new SimpleDateFormat("yyyy-MM-dd hh:mm:ss").format(new Date()));
					try {
						Thread.sleep(5000);//模拟每隔线程执行5s
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
				}
			});
		}
		System.out.println("---------------END--------------");
	}


	/**
	 * 描述:
	 * 		结果依次输出,相当于顺序执行各个任务
	 * result:
	 * ---------------BEGIN--------------
	 * 当前执行任务index值为:1
	 * 当前执行任务index值为:2
	 * 当前执行任务index值为:3
	 * 当前执行任务index值为:4
	 * 当前执行任务index值为:5
	 * 当前执行任务index值为:6
	 * 当前执行任务index值为:7
	 * 当前执行任务index值为:8
	 * 当前执行任务index值为:9
	 * ---------------END--------------
	 */
	public static void singleMethod() {
		ExecutorService pool = ThreadPoolUtil.singleInstance();
		System.out.println("---------------BEGIN--------------");
		for (int i = 1; i < 10; i++) {
			final  int index = i;
			pool.execute(new Runnable() {
				public void run() {
					System.out.println("当前执行任务index值为:"+index);
				}
			});
			try {
				Thread.sleep(2000);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
		System.out.println("---------------END--------------");
	}
}

    
    
----------->BEGIN<-------