Java 线程池

1. 线程

1.1 线程的几种状态:

  • NEW:初始状态、线程被构建,但是还没有 start()
  • RUNNABLE:运行状态,RUNINGREADY 都是运行状态,start() 之后变成 READY 线程获取 CPU 的 timeslice 之后就处于 RUNNING
  • BLOCK:阻塞状态,线程阻塞于锁
  • WAITING:等待状态,需要等待其它线程做出一些动作
  • TIME_WAITING:超时等待,到达指定的时间自行返回
  • TERMINATE:线程终止,已经执行完毕了
    在这里插入图片描述

1.2 sleepwait 的区别

  • sleep 没有释放锁,而 wait 释放了锁
  • wait 主要和 notify 配合使用来进行线程间的通信,sleep 主要是用于暂停线程

1.3 线程创建的几种方式

  • 继承 Thread,重写 run 方法
  • 实现 Runnable 方法,使用 Thread 的构造函数
  • 使用 CallableFutureTask 创建线程
Callable<String> callable = new Thread3();
FutureTask<String> futureTask = new FutureTask<>(callable);
Thread thread3 = new Thread(futureTask);
  • 使用线程池
Executor executor = Executors.newFixedThreadPool(5);
executor.execute(new Runnable() {
	@Override
	public void run() {
		System.out.println(Thread.currentThread()+"创建线程的第四种方法");
	}
});

2. Executor

组成

  • Runnable / Callable:任务,其接口的实现类都可以被 ThreadPoolExecutorScheduledThreadPoolExecutor 执行
  • Executor:任务的执行
  • Future:异步计算的结果,调用 submit 时返回的对象

流程

  • 主线程创建实现 Runnable 或者 Callable 接口的任务对象
  • 把创建完成的实现接口的对象直接交给 ExecutorService 执行,execute()submit()
  • 如果 submit,会返回一个 Future 的实现类对象
  • 最后主线程执行返回的 Future 对象的 get 方法来等待任务执行完成,主线程也可以执行 cancel 方法来取消任务的执行

3. ThreadPoolExecutor

阿里巴巴推荐使用 ThreadPoolExecutor 构建线程池

3.1 主要参数

  • corePoolSize:核心线程数,最小同时运行线程数量
  • maximumPoolSize:最大线程数,队列达到容量时,可同时运行的线程数量变成最大线程数
  • workQueue:工作队列,当前运行线程到达核心线程数的时候,任务进入工作队列
  • keepAliveTime:当线程数大于核心线程数时,多余的空闲线程存活的最长时间(没有新任务提交的时候)
  • unitkeepAliveTime 参数的时间单位
  • threadFactory:线程工厂,用来创建线程,一般默认即可
  • handler:拒绝策略,当提交的任务过多而不能及时处理时,我们可以定制策略来处理任务

3.2 拒绝策略

线程数量达到最大线程数,队列也满了的时候

  • AbortPolicy:默认的拒绝策略,直接抛出 RejectedExecutionException
  • CallerRunsPolicy:由调用线程(提交任务的线程)处理该任务,如果执行程序已关闭,则会丢弃该任务。因此这种策略会降低对于新任务提交速度,影响程序的整体性能。如果您的应用程序可以承受此延迟并且你要求任何一个任务请求都要被执行的话,你可以选择这个策略。
  • DiscardPolicy:不处理新任务,直接丢弃掉
  • DiscardOldestPolicy:丢弃最早的未处理的任务请求

3.3 Demo

public class Demo {

    private static final int CORE_POOL_SIZE = 5;
    private static final int MAX_POOL_SIZE = 10;
    private static final int QUEUE_CAPACITY = 100;
    private static final Long KEEP_ALIVE_TIME = 1L;

    public static void main(String[] args) {

        ThreadPoolExecutor executor = new ThreadPoolExecutor(
                CORE_POOL_SIZE,
                MAX_POOL_SIZE,
                KEEP_ALIVE_TIME,
                TimeUnit.SECONDS,
                new ArrayBlockingQueue<>(QUEUE_CAPACITY),
                new ThreadPoolExecutor.CallerRunsPolicy());

        List<Future<String>> futureList = new ArrayList<>();
        Callable<String> callable = new MyCallable();
        for (int i = 0; i < 10; i++) {
            //提交任务到线程池
            Future<String> future = executor.submit(callable);
            //将返回值 future 添加到 list,我们可以通过 future 获得 执行 Callable 得到的返回值
            futureList.add(future);
        }
        for (Future<String> fut : futureList) {
            try {
                System.out.println(new Date() + "::" + fut.get());
            } catch (InterruptedException | ExecutionException e) {
                e.printStackTrace();
            }
        }
        //关闭线程池
        executor.shutdown();
    }
}
Wed Jul 01 21:51:35 CST 2020::pool-1-thread-1
Wed Jul 01 21:51:36 CST 2020::pool-1-thread-2
Wed Jul 01 21:51:36 CST 2020::pool-1-thread-3
Wed Jul 01 21:51:36 CST 2020::pool-1-thread-4
Wed Jul 01 21:51:36 CST 2020::pool-1-thread-5
Wed Jul 01 21:51:36 CST 2020::pool-1-thread-3
Wed Jul 01 21:51:37 CST 2020::pool-1-thread-1
Wed Jul 01 21:51:37 CST 2020::pool-1-thread-2
Wed Jul 01 21:51:37 CST 2020::pool-1-thread-5
Wed Jul 01 21:51:37 CST 2020::pool-1-thread-4

4. 常见的线程池

4.1 FixedThreadPool

public static ExecutorService newFixedThreadPool(int nThreads, ThreadFactory threadFactory) {
    return new ThreadPoolExecutor(nThreads, nThreads,
                                  0L, TimeUnit.MILLISECONDS,
                                  new LinkedBlockingQueue<Runnable>(),
                                  threadFactory);
}
  • 核心线程数和最大线程数被设置成了同一个值
  • 使用无界队列作为工作队列 LinkedBlockingQueue,队列不会被装满
  • 所以不会拒绝任务,可能会有 OOM

4.2 SingleThreadExecutor

public static ExecutorService newSingleThreadExecutor(ThreadFactory threadFactory) {
    return new FinalizableDelegatedExecutorService
        (new ThreadPoolExecutor(1, 1,
                                0L, TimeUnit.MILLISECONDS,
                                new LinkedBlockingQueue<Runnable>(),
                                threadFactory));
}
  • 返回只有一个线程的线程池
  • 同样使用的也是无界队列

4.3 CachedThreadPool

public static ExecutorService newCachedThreadPool(ThreadFactory threadFactory) {
    return new ThreadPoolExecutor(0, Integer.MAX_VALUE,
                                  60L, TimeUnit.SECONDS,
                                  new SynchronousQueue<Runnable>(),
                                  threadFactory);
}
  • 可能会创建大量的线程,极端情况下会 OOM
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值