1、newCachedThreadPool

创建一个可根据需要创建新线程的线程池,但是在以前构造的线程可用时将重用他们。对于执行很多短期异步任务的程序而言,这些线程池可提高程序性能。调用execute将重用以前构造的线程(如果有线程可用)。如果没有线程可用,则创建一个新线程并添加到池中。线程池会终止并从缓存中移除那些已有60秒钟未被使用的线程。因此,长时间保持空闲的线程池不会使用任何资源。

public static void CacheThreadPool(){
    ExecutorService cachedThreadPool = Executors.newCachedThreadPool();
    for(int i=0;i<10;i++){
        int index = i;
        try{
            Thread.sleep(index*1000);
        }catch (InterruptedException e){
            e.printStackTrace();
        }
        cachedThreadPool.execute(new Runnable() {
            @Override
            public void run() {
                System.out.println(index);
            }
        });
    }
}

2、newFixedThreadPool

创建一个可重用的固定线程数的线程池(指定线程的最大并发数),以共享的无界队列方式来运行这些线程。在任意点,在大多数nThreads线程会处于处理任务的活动状态。如果在所有线程处于活动状态时提交附加任务,则在有可用线程之前,附加任务将在队列中等待。

  • 如果在关闭前的执行期间由于失败而导致任何线程终止,那么一个新线程将代替他执行后续的任务(如果需要)。
  • 在某个线程被显式地关闭之前,池中的线程将一直存在。
public static void main(String[] args){
    ExecutorService fixedThreadPool = Executors.newFixedThreadPool(3);
    for (int i = 0; i < 10; i++) {
        final int index = i;
        fixedThreadPool.execute(new Runnable() {
            @Override
            public void run() {
                try {
                    System.out.println(index);
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });
    }
}

3、newScheduledThreadPool

创建一个线程池,它可安排在给定延迟后运行命令或者定期地执行。

public static void main(String[] args){
    ScheduledExecutorService scheduledThreadPoo l= Executors.newScheduledThreadPool(3);
    scheduledThreadPool.schedule(newRunnable() { 
        @Override 
        public void run() {
            System.out.println("延迟三秒");
        }
    }, 3, TimeUnit.SECONDS);

    scheduledThreadPool.scheduleAtFixedRate(newRunnable(){ 
        @Override 
        public void run() {
            System.out.println("延迟 1 秒后每三秒执行一次");
        }
    }, 1, 3, TimeUnit.SECONDS);
}

4、newSingleThreadExecutor

Executors.newSingleThreadExecutor()返回一个线程池(这个线程池只有一个线程),这个线程池可以在线程死后(或发生异常时)重新启动一个线程来替代原来的线程继续执行下去。

理解:按照顺序执行等待队列中的任务,挂了一个线程会新开一个线程继续执行线程池中的任务!

可以使用这个做一个保活程序(https://blog.csdn.net/weixin_40693633/article/details/82958046)双守护进程

public static void main(String[] args){
    ExecutorService singleThreadExecutor = Executors.newSingleThreadExecutor();
    for (int i = 0; i < 10; i++) {
        final int index = i;
        singleThreadExecutor.execute(new Runnable() {

            @Override
            public void run() {
                try {
                    System.out.println(index);
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });
    }
}