zoukankan      html  css  js  c++  java
  • 线程学习四:线程池

    Java通过Executors提供四种线程池,分别为:

    • newCachedThreadPool创建一个可缓存线程池,如果线程池长度超过处理需要,可灵活回收空闲线程,若无可回收,则新建线程。
    • newFixedThreadPool 创建一个定长线程池,可控制线程最大并发数,超出的线程会在队列中等待。
    • newScheduledThreadPool 创建一个定长线程池,支持定时及周期性任务执行。
    • newSingleThreadExecutor 创建一个单线程化的线程池,它只会用唯一的工作线程来执行任务,保证所有任务按照指定顺序(FIFO, LIFO, 优先级)执行。

    一、newCachedThreadPool

     1 public class NewCachedThreadPool {
     2     public static void main(String[] args) throws InterruptedException {
     3         ExecutorService es = Executors.newCachedThreadPool();
     4         RunnableTest1 rt = new RunnableTest1();
     5         es.execute(rt);
     6         Thread.sleep(2000);
     7         System.out.println("sleep finish");
     8         es.execute(rt);
     9     }
    10 }
    11 
    12 class RunnableTest1 implements Runnable {
    13     public void run() {
    14         for (int i = 0; i < 100; i++) {
    15             System.out.println(Thread.currentThread().getName() + "...." + i);
    16         }
    17     }
    18 }

    如果上一个任务已经使用线程完毕,下一个任务会调用之前的线程,不会再创建;如果任务没有完毕,会创建新线程。

    二、newFixedThreadPool

     1 public class NewFixedThreadPool {
     2     public static void main(String[] args) throws InterruptedException {
     3         ExecutorService es = Executors.newFixedThreadPool(2);
     4         RunnableTest rt = new RunnableTest();
     5         es.execute(rt);
     6         Thread.sleep(2000);
     7         System.out.println("sleep finish");
     8         es.execute(rt);
     9     }
    10 }
    11 
    12 class RunnableTest implements Runnable {
    13     public void run() {
    14         for (int i = 0; i < 100; i++) {
    15             System.out.println(Thread.currentThread().getName() + "...." + i);
    16         }
    17     }
    18 }

    规定线程池里面最大线程数,定长线程池的大小最好根据系统资源进行设置。如Runtime.getRuntime().availableProcessors()。

    三、newScheduledThreadPool

     1 public class NewScheduledThreadPool {
     2     public static void main(String[] args) {
     3         ScheduledExecutorService es = Executors.newScheduledThreadPool(2);
     4         RunnableTest2 rt = new RunnableTest2();
     5         es.scheduleAtFixedRate(rt, 0, 3, TimeUnit.SECONDS);
     6         es.execute(rt);
     7     }
     8 }
     9 
    10 class RunnableTest2 implements Runnable {
    11     public void run() {
    12         for (int i = 0; i < 100; i++) {
    13             System.out.println(Thread.currentThread().getName() + "...." + i);
    14         }
    15     }
    16 }

    可以设置线程的延迟时间以及隔多久执行一次,注意这里是调用ScheduledExecutorService的scheduleAtFixedRate()方法,与其他线程池不同。

    四、newSingleThreadScheduledExecutor

     1 public class NewSingleThreadExecutor {
     2     public static void main(String[] args) {
     3         ExecutorService es = Executors.newSingleThreadScheduledExecutor();
     4         RunnableTest3 rt = new RunnableTest3();
     5         es.execute(rt);
     6         es.execute(rt);
     7     }
     8 }
     9 
    10 class RunnableTest3 implements Runnable {
    11     public void run() {
    12         for (int i = 0; i < 100; i++) {
    13             System.out.println(Thread.currentThread().getName() + "...." + i);
    14         }
    15     }
    16 }

    会将并行的线程按一定排序串行执行。

    五、线程池关闭

    我们可以通过调用线程池的shutdown或shutdownNow方法来关闭线程池,它们的原理是遍历线程池中的工作线程,然后逐个调用线程的interrupt方法来中断线程,所以无法响应中断的任务可能永远无法终止。但是它们存在一定的区别,shutdownNow首先将线程池的状态设置成STOP,然后尝试停止所有的正在执行或暂停任务的线程,并返回等待执行任务的列表,而shutdown只是将线程池的状态设置成SHUTDOWN状态,然后中断所有没有正在执行任务的线程。

    只要调用了这两个关闭方法的其中一个,isShutdown方法就会返回true。当所有的任务都已关闭后,才表示线程池关闭成功,这时调用isTerminaed方法会返回true。至于我们应该调用哪一种方法来关闭线程池,应该由提交到线程池的任务特性决定,通常调用shutdown来关闭线程池,如果任务不一定要执行完,则可以调用shutdownNow。

  • 相关阅读:
    面试题:找出数组中只出现一次的2个数(异或的巧妙应用)(出现3次)
    线段树 | 第1讲 (给定区间求和)(转)
    C++中的静态多态和动态多态(转)
    ARP与RARP详细解析(转)
    排序算法之归并排序
    byte数组使用Arrays.asList转换List出错
    排序算法之希尔排序
    排序算法之冒泡排序、选择排序、插入排序
    Tomcat配置优化
    内连接、左外连接、右外连接、全外连接、交叉连接
  • 原文地址:https://www.cnblogs.com/gforce/p/5909691.html
Copyright © 2011-2022 走看看