关闭

java 线程池和队列的小研究

587人阅读 评论(0) 收藏 举报
分类:

有时候项目中要使用到队列和多线程,写个例子留着随笔,方便后续使用

大家都知道,多线程是为了能更高效的运行程序,而线程池是为了控制一个进程中线程过多而导致内存溢出的问题,队列主要是为了解决了某一时刻请求过多而出现的宕机情况,同时能接收处理更多的请求,可以将多个请求放置到队列中,等待线程依次执行,而客户端则可以继续处理其它事情。


线程池和队列执行顺序:

 1、线程池刚创建时,里面没有一个线程。任务队列是作为参数传进来的,不过就算队列里面有任务,线程池也不会马上执行它们。

 2、当调用execute() 方法添加一个任务时,线程池会做如下判断:

        a、如果正在运行的线程数量小于corePoolSize,那么马上创建线程运行这个任务;

        b、如果正在运行的线程数量大于或等于corePoolSize,那么将这个任务放入队列;

        c、如果这时候队列满了,而且正在运行的线程数量小于maximumPoolSize,那么还是要创建新的线程运行这个任务;

        d、如果队列满了,而且正在运行的线程数量大于或等于maximumPoolSize,那么线程池会抛出异常,告诉调用者“我不能再接收任务”了;

 3、当一个线程完成时,它会从队列中取下一个任务来执行。

 4、当一个线程无事可做,超过一定的时间时,线程池会判断,如果当前运行的线程数大于corePoolSize,那么这个线程被停掉,所以线程池的所有任务完成后,           它会收缩到corePoolSzie 的大小。

 这个过程说明,并不是先加入任务就一定会执行。假设队列大小为4,corePoolSize为2,maximumPoolSize为6,那么当加入15个任务时,执行的顺序类似这样:

        首先执行任务1、2,然后任务3~6被放入队列,这个时候队列满了,任务7、8、9、10会被马上执行,而任务11~15则会抛出异常。最终的顺序是:1、2、7、8、9、10、3、4、5、6。当然这个过程是针对指定大小的ArrayBlockingQueue<Runnable> 来说,如果是LinkedBlockingQueue<Runnable>,因为该队列无带下限制,所以不存在上述问题。



1、线程池和队列的应用小例子

       (1) 线程实现类

      public class ThreadTest implements Runnable {
              @Override
              public void run() {
                 /** 设置获取当前线程的名称 **/
                 System.out.println(Thread.currentThread().getName()); 
                  try{
                         /** 设置当前线程休眠3秒钟 **/
                         Thread.sleep(3000);
                     } catch(InterruptedException e) {
                        e.printStackTrace(); 
                    }
              }
       }
       (2) 主线程类

<pre name="code" class="java">      public static void main(String[] args ) throws InterruptedException {
           /** 创建一个线程队列,不指定队列大小,则无大小限制,队列存放线程超类 **/
            BlockinQueue<Runnable> queue = new LinkedBlockingQueue<Runnable>();
           /** 创建线程池对象,线程池根据参数创建指定个数的线程, 这里2指:空闲线程个数,6是指:最大线程个数,1是指:一个线程不工作时,超过该时间就**/             /** 就停止该线程,TimeUnit.DAYS是指:时间的单位,queue是指:存放需要被线程池执行的线程队列 **/
            ThreadPoolExecuter executor = new ThreadPoolExecutor(2,6,1,TimeUnit.DAYS,queue); 
            for(int i=0;i<10;i++) { 
              /** 设定每一个线程的名称 **/ 
              executor.execute(new Thread(new ThreadTest(),"ThreadName-"+i)); 
              System.out.println("线程队列大小为---->"+queue.size()); 
            }
            executor.shutdown(); 
      }


</pre><p></p><pre>
     (3) 输出结果

          



1
0

查看评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
    个人资料
    • 访问:15643次
    • 积分:289
    • 等级:
    • 排名:千里之外
    • 原创:13篇
    • 转载:3篇
    • 译文:0篇
    • 评论:4条
    文章分类
    最新评论