25_线程池_几个常见的线程池

【线程池简述】java

线程池中,当须要使用线程时,会从线程池中获取一个空闲线程,线程完成工做时,不会直接关闭线程,而是将这个线程退回到池子,方便其它人使用。框架

简而言之,使用线程池后,原来建立线程变成了从线程池得到空闲线程,关闭线程变成了向池子归还线程。ide

 

【线程池带来的好处】性能

1.下降资源消耗,经过重复利用已建立的线程下降线程建立和销毁形成的性能消耗。ui

2.提升响应速度,当任务到达时,任务能够不须要等待线程建立,能够直接执行。spa

3.提升线程的可管理性,线程是稀缺资源,若是无限制地建立,不只会消耗系统资源,还会下降系统的稳定性,使用线程池能够进行统一的分配,调优和监控。线程

 

【线程池接口、类关系一览】 code

 

【说明】对象

Executor是一个顶级接口,它里面只声明一个方法:execute(Runnable command),用来执行传进去的任务。blog

ExecutorService接口继承了Executor接口,并声明了一些方法:submit、shutdown、invokeAll等。

AbstractExecutorService抽象类实现了ExecutorService接口,基本实现了ExecutorService接口的全部方法。

ThreadPoolExecutor继承了类AbstractExecutorService。

 

【Executors框架】

Executors框架提供了各类类型的线程池,主要有如下几种工厂方法:

[ newFixedThreadPool()方法 ]

该方法返回一个固定线程数量的线程池。该线程池中的线程数量始终不变,当有一个新的任务提交时,线程池中如有空闲线程,则当即处理。

若没有空闲线程,则新的任务会被暂存在一个任务队列中,待有线程空闲时,便处理在任务队列中的任务。

[ newSingleThreadExecutor()方法 ]

该方法返回一个只有一个线程的线程池。

若多多余一个任务被提交到该线程池,任务会被保存在一个任务队列中,待线程空闲,按先入先出顺序执行任务。

[ newCachedThreadPool()方法 ]

该方法返回一个可具实际状况调整线程数量的线程池,线程池的线程数量不肯定,但如有空闲线程能够复用,则会有优先使用而能够复用线程。若全部线程均在工做,又有新的任务提交,则会建立新的现场处理任务。全部线程在当前任务执行完毕后,将返回线程池进行复用。

[ newSingleThreadScheduledExecutor()方法 ]

该方法返回一个ScheduleExecutorService对象,线程池大小为1

ScheduleExecutorService接口在ExecutorService接口之上扩张了在给定时间执行某任务的功能,如:在某个固定的延时以后执行,或者周期性执行某个任务。

[ newScheduleThreadPool()方法 ]

该方法返回一个ScheduleExecutorService对象,但该线程池能够指定线程数量

 

【固定大小的线程池——newFixedThreadPool()】

package com.test.executor;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class ExecutorDemo1 {
    
    public static void main(String[] args) {
        MyTask task = new MyTask();
        ExecutorService es = Executors.newFixedThreadPool(5);  //建立固定线程数大小为5的线程池
        for(int i=0;i<10;i++){   //依次向线程池提交了10个任务
            es.submit(task);
        }
    }
}

class MyTask implements Runnable{

    @Override
    public void run() {
        System.out.println(System.currentTimeMillis()+":Thread ID:"+Thread.currentThread().getId());
        try{
            Thread.sleep(1000); //1秒
        }catch(Exception e){
            e.printStackTrace();
        }
    }
    
}

【运行结果】

 

【只有一个线程的线程池——newSingleThreadExecutor()】 

package com.test.executor;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

/**
 * Created by HigginCui on 2018/4/14.
 */
public class ExecutorDemo1 {

    public static void main(String[] args) {
        MyTask myTask = new MyTask();
        //只有一个线程的线程池
        ExecutorService es = Executors.newSingleThreadExecutor();
        for(int i=0;i<10;i++){
            es.submit(myTask);
        }
    }
}

class MyTask implements Runnable{
    @Override
    public void run() {
        System.out.println(System.currentTimeMillis()/1000 + ":Thread ID:" + Thread.currentThread().getId());
        try {
            Thread.sleep(1000);
        }catch (Exception e){
            e.printStackTrace();
        }
    }
}

【运行结果】

 

【可根据实际状况调整线程数量的线程池——newCacheThreadPool()】

package com.test.executor;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

/**
 * Created by HigginCui on 2018/4/14.
 */
public class ExecutorDemo1 {

    public static void main(String[] args) {
        MyTask myTask = new MyTask();
        //可根据实际状况调整线程数量的线程池
        ExecutorService es = Executors.newCachedThreadPool();
        for(int i=0;i<10;i++){
            es.submit(myTask);
        }
    }
}

class MyTask implements Runnable{
    @Override
    public void run() {
        System.out.println(System.currentTimeMillis()/1000 + ":Thread ID:" + Thread.currentThread().getId());
        try {
            Thread.sleep(1000);
        }catch (Exception e){
            e.printStackTrace();
        }
    }
}

【运行结果】

 

【计划定时任务——newScheduledThreadPool】

package com.test.executor;

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

/**
 * Created by HigginCui on 2018/4/14.
 */
public class ExecutorDemo1 {

    public static void main(String[] args) {

        ScheduledExecutorService ses = Executors.newScheduledThreadPool(10);
        /**
         * scheduleAtFixedRate方法 :若是前面的任务没有完成,则调度也不会执行!
         * scheduleAtFixedRate(Runnable command, long initialDelay, long period, TimeUnit unit);
         */
        ses.scheduleAtFixedRate(new ScheduledTimeTask(), 0, 2, TimeUnit.SECONDS);  //设置每定时2s执行一次
    }

}

class ScheduledTimeTask implements Runnable {
    @Override
    public void run() {
        try {
            //修改这里的任务执行时间
            Thread.sleep(1000);
            System.out.println( System.currentTimeMillis() / 1000 + " : ThreadId = " + Thread.currentThread().getId());
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

【设置任务的执行时间为1s( <定时的2s )的运行结果】

【设置任务的执行时间为3s( <定时的2s )的运行结果(即代码改为Thread.sleep(3000))】

相关文章
相关标签/搜索