JAVA编码(42)—— 线程池队列执行任务(ThreadPoolQueue)(2)

1. ThreadPoolExecutor的一个常用的构造方法

 

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

 

参数说明:

-corePoolSize       线程池中所保存的核心线程数。线程池启动后默认是空的,只有任务来临时才会创建线程以处理请求。prestartAllCoreThreads方法可以在线程池启动后即启动所有核心线程以等待任务。

-maximumPoolSize  线程池允许创建的最大线程数。当workQueue使用无界队列时(如:LinkedBlockingQueue),则此参数无效。

-keepAliveTime      当前线程池线程总数大于核心线程数时,终止多余的空闲线程的时间。

-unit          keepAliveTime参数的时间单位。

-workQueue       工作队列,如果当前线程池达到核心线程数时(corePoolSize),且当前所有线程都处于活动状态,则将新加入的任务放到此队列中。下面仅列几个常用的:

  • ArrayBlockingQueue:  基于数组结构的有界队列,此队列按FIFO原则对任务进行排序。如果队列满了还有任务进来,则调用拒绝策略。
  • LinkedBlockingQueue:  基于链表结构的无界队列,此队列按FIFO原则对任务进行排序。因为它是无界的,根本不会满,所以采用此队列后线程池将忽略拒绝策略(handler)参数;同时还将忽略最大线程数(maximumPoolSize)等参数。
  • SynchronousQueue:   直接将任务提交给线程而不是将它加入到队列,实际上此队列是空的。每个插入的操作必须等到另一个调用移除的操作;如果新任务来了线程池没有任何可用线程处理的话,则调用拒绝策略。其实要是把maximumPoolSize设置成无界(Integer.MAX_VALUE)的,加上SynchronousQueue队列,就等同于Executors.newCachedThreadPool()。
  • PriorityBlockingQueue: 具有优先级的队列的有界队列,可以自定义优先级;默认是按自然排序,可能很多场合并不合适。

-handler          拒绝策略,当线程池与workQueue队列都满了的情况下,对新加任务采取的策略。

  • AbortPolicy:           拒绝任务,抛出RejectedExecutionException异常。默认值。
  • CallerRunsPolicy:   A handler for rejected tasks that runs the rejected task directly in the calling thread of the execute method, unless the executor has been shut down, in which case the task is discarded(没太弄懂意思,看不太懂,程序模拟半天也没得出啥结论。)
  • DiscardOldestPolicy:  如果执行程序尚未关闭,则位于工作队列头部的任务将被删除,然后重试执行程序(如果再次失败,则重复此过程)。这样的结果是最后加入的任务反而有可能被执行到,先前加入的都被抛弃了。
  • DiscardPolicy:      加不进的任务都被抛弃了,同时没有异常抛出。

 

2. 详解及示范

    1. 一个任务进来(Runnable)时,如果核心线程数(corePoolSize)未达到,则直接创建线程处理该任务;如果核心线程数已经达到则该任务进入工作队列(workQueue)。如果工作队列满了(只能是有界队列),则检查最大线程数(maximumPoolSize)是否达到,如果没达到则创建线程处理任务(FIFO);如果最大线程数据也达到了,则调用拒绝策略(handler)。
    2. 如果workQueue使用LinkedBlockingQueue队列,因为它是无界的,队列永远不会满,所以maximumPoolSize参数是没有意义的,同样keepAliveTimeunithandler三个参数都无意义。
    3. 如果workQueue使用ArrayBlockingQueue队列,那么小心,因为此队列是有界的,必须小心处理拒绝策略。你看人家Executors类,压根就不使用ArrayBlockingQueue队列。
    4. 正常情况下,如果使用Executors静态工厂生成的几种常用线程池能够满足要求,建议就用它们吧。自己控制所有细节挺不容易的。

废话不说,直接上代码

package com.sinosoft;

import java.util.concurrent.*;

/**
 * Created by xushuyi on 2017/4/9.
 */
public class ThreadPool {

    /**
     * 线程池中所保存的核心线程数
     */
    private static final Integer COREPOOLSIZE = 5;

    /**
     * 线程池允许创建的最大线程数
     */
    private static final Integer MAXIMUMPOOLSIZE = 20;

    /**
     * 当前线程池线程总数大于核心线程数时,终止多余的空闲线程的时间
     */
    private static final Integer KEEPALIVETIME = 5;

    /**
     * 创建一个任务队列
     */
    private static final BlockingQueue<Runnable> QUEUE = new LinkedBlockingQueue<Runnable>();

    /**
     * 创建一个线程池
     */
    protected static ThreadPoolExecutor threadPool = new ThreadPoolExecutor(2, 4, 5, TimeUnit.SECONDS, QUEUE, new ThreadPoolExecutor.AbortPolicy());

    static class thread implements Runnable {

        private String threadName;

        thread(String thread) {
            this.threadName = thread;
        }

        /**
         * 执行任务
         */
        public void run() {
            //开始执行任务
            try {
                Thread.sleep(1000);

                System.out.println(threadName + " finished job!");
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }

    public static void main(String[] args) {
        //往线程池中放任务
        for (int i = 0; i < 10; i++) {
            System.out.println("当前线程池大小[" + threadPool.getPoolSize() + "],当前队列大小[" + QUEUE.size() + "]");
            threadPool.execute(new thread("Thread" + i));
        }
        //关闭线程池
        threadPool.shutdown();
    }
}

3. 回头看看Executors静态工厂方法生成线程池的源代码

Executors.newSingleThreadExecutor()

public static ExecutorService newSingleThreadExecutor() {
        return new FinalizableDelegatedExecutorService
            (new ThreadPoolExecutor(1, 1,
                                    0L, TimeUnit.MILLISECONDS,
                                    new LinkedBlockingQueue<Runnable>()));
    }

Executors.newFixedThreadPool()

    public static ExecutorService newFixedThreadPool(int nThreads) {
        return new ThreadPoolExecutor(nThreads, nThreads,
                                      0L, TimeUnit.MILLISECONDS,
                                      new LinkedBlockingQueue<Runnable>());
    }

Executors.newCachedThreadPool()

    public static ExecutorService newCachedThreadPool() {
        return new ThreadPoolExecutor(0, Integer.MAX_VALUE,
                                      60L, TimeUnit.SECONDS,
                                      new SynchronousQueue<Runnable>());
    }

通过上述代码可以发现,用Executors静态工厂生成的几种常用线程池,都可以向里面插入n多任务:要么workQueue是无界的,要么maximumPoolSize是无界的。

posted @ 2017-04-09 21:25  xu_shuyi  阅读(179)  评论(0)    收藏  举报