线程池与Android的日日夜夜
假如你Java中研究到了线程池的话,一般来说,你已经对线程的原理颇有研究了,或者说,你意识到了线程的某些瓶颈或者缺点。你说,要有光,所以,天降线程池。
正儿八经的说,如果你为每一个请求创建一个新的线程,这在性能上影响是巨大的,因为线程对象的创建销毁需要Java虚拟机频繁的GC,假如说,一个请求所用的时间比创建销毁线程对象时间还短的话,那么时间将会大程度浪费在虚拟机的GC上,系统性能降低。
所以啊,线程池主要就是复用线程对象,就跟上面所说,解决线程对象频繁创建和销毁的问题,内部可以抽象成一个“池”,线程对象放在里头,需要用的时候就拿出来用,不用了就泡着,泡坏了或者不要了就清掉。也正因为如此,线程池可以用来处理高并发的访问请求。
目录
- 先从最基本的线程的3种用法说起
- 一个最基本的线程池用例
- 分析各种参数:线程池创建的ThreadPoolExecutor类
- 常见阻塞队列及使用场景
- 比较Executors中3种线程池的区别和使用情景
- 对比线程和线程池的优缺点,各种使用场景及其区别
- 其他:并发集合框架
- 默认Executors生成线程池和自传参数进构造方法ThreadPoolExecutor创建线程池的利弊
- 分析实际应用,如OkHttp中的线程池,如AsyncTask中的线程池,RxJava中的线程池
一 先从最基本的线程开始
先重新了解一下,创建线程的三种方法:
- 继承Thread类创建线程
- 实现Runnable创建线程
- 实现Callable接口 、使用Future类接收返回值
(1)继承Thread类,重写父类run方法
public class MyThread extends Thread { |
(2)实现Runnable接口,实现接口的run方法
public class MyRunnable implements Runnable { |
当然,我们最常用的是匿名的内部Runnable类
public class MyRunnable { |
(3)实现Callable接口,使用Future来接收返回值(接收可选)
import java.util.concurrent.Callable; |
二、一个最基本的线程池用例
先放一个基本的线程池,这里构造的是核心线程为2,最大线程数为5,有界阻塞数列为5的线程池
import java.util.concurrent.*; |
console如下
Exception in thread "main" java.util.concurrent.RejectedExecutionException: Task MyDemo$1@135fbaa4 rejected from java.util.concurrent.ThreadPoolExecutor@45ee12a7[Running, pool size = 5, active threads = 5, queued tasks = 5, completed tasks = 0] |
先看这里的console,这里打印了
# 三、线程池创建的ThreadPoolExecutor类
ThreadPoolExecutor类有4个构造方法,其中的三个构造方法最终会调用参数最多的(7个)的构造方法
public ThreadPoolExecutor(int corePoolSize,
int maximumPoolSize,
long keepAliveTime,
TimeUnit unit,
BlockingQueue
ThreadFactory threadFactory,
RejectedExecutionHandler handler) {
..
|
```pool-1-thread-1```,一般默认为```Executors.defaultThreadFactory()```即可,当然,ThreadPoolExecutor类的构造方法最终都是传入```DefaultThreadFactory``` 7. ```RejectedExecutionHandler```:饱和策略。当任务队列和线程池都达到最大值时的处理策略。默认是无法处理新的任务的```AbortPolicy```。比如上面第二节console输出的是```AbortPolicy```策略。那是因为创建的最大线程数是5,任务队列是5,那么线程池中会存在10个线程,而我创建了13个线程的同时超过了10个线程,接着就会抛出这个```RejectedExecutionHandler```异常
###### 1. 线程池的处理过程
拿第二节创建的线程池来举例,核心线程是2,最大线程数是5(说明非核心线程数为3)。任务队列是ArrayBlockingQueue(特点是它用数组实现,元素排序规则是先进先出,默认不保证线程池按照阻塞的先后顺序访问队列),数量为5个,超时为6秒,其他都为默认。
那么,其实内部是这样的-------->看图:
1. 核心线程未饱和
![核心线程未饱和.gif](https://upload-images.jianshu.io/upload_images/3515789-b4a321c33910abec.gif?imageMogr2/auto-orient/strip)
当只有1核心线程时,这时新建的任务会直接添加为核心线程
---
2. 核心线程饱和队列未饱和
![核心线程饱和队列未饱和.gif](https://upload-images.jianshu.io/upload_images/3515789-82e987f0ca11a208.gif?imageMogr2/auto-orient/strip)
当核心线程已满,任务队列未饱和时,这时新建的任务会添加到工作队列
---
3. 核心线程饱和队列饱和非核心线程未饱和
![核心线程饱和队列饱和非核心线程未饱和.gif](https://upload-images.jianshu.io/upload_images/3515789-471b1c192a834240.gif?imageMogr2/auto-orient/strip)
当核心线程已满,任务队列已饱和,非核心线程未饱和时,新建的任务会添加为非核心线程。
---
4. 核心线程饱和队列饱和非核心线程饱和
![核心线程饱和队列饱和非核心线程饱和.gif](https://upload-images.jianshu.io/upload_images/3515789-2fae49209fbdea7a.gif?imageMogr2/auto-orient/strip)
当线程池线程已达最大值,队列也已饱和,这时新建任务会执行饱和策略
---
**总结起来,其实就是:**
![线程池执行流程.png](https://upload-images.jianshu.io/upload_images/3515789-83b471856880e85f.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
---
这里很懵逼的是阻塞队列,事实上不是每个阻塞队列都像```ArrayBlockingQueue```如此,下节将分析常用的阻塞队列
# 四、常见阻塞队列及使用场景
阻塞队列使用方法大同小异,只要了解他的内部结构构成,以及由其结构影响的各种特性即可,具体测试及用法可看
[BlockingQueue(阻塞队列)详解](https://www.cnblogs.com/tjudzj/p/4454490.html)、
[深入理解阻塞队列(二)——ArrayBlockingQueue源码分析](https://blog.csdn.net/qq_19431333/article/details/72864017)、
[深入理解阻塞队列(三)——LinkedBlockingQueue源码分析](https://blog.csdn.net/qq_19431333/article/details/73087366)
常用的几种阻塞队列如下:
- ArrayBlockingQueue:有界阻塞队列,它用数组实现,**元素排序规则是先进先出**,默认不保证线程池按照阻塞的先后顺序访问队列,一般构造方法会指定元素数量,和是否公平顺序按照阻塞顺序访问队列,通常用在需要生产者和消费者顺序的操作队列中的数据,以降低吞吐量的时候,比如
- LinkedBlockingQueue:有界阻塞队列,链表实现,与ArrayBlockingQueue区别不同,**它是并行的操作队列中的数据**,这也决定了它能用于高并发,巨大吞吐量的情况,需要注意的是,LinkedBlockingQueue的容量记得要指定哦,不然太大了加入生产者的速度大于消费者,那么队列阻塞可能不会阻塞,因为内存会炸。
- SyschronousQueue:**不存储元素的异步队列,有个特点,插入操作的完成要等待另一个线程的对应移除操作**,适合那种立即处理且耗时较少的任务。
# 五、比较Executors中3种线程池的区别和使用情景
其实不止3种(有6种),这里只分析其中常用的4种,其他的大同小异
1. ```FixedThreadPool```:固定线程数的线程池,特点在核心线程和线程最大数量相等,意味着只有核心线程,keepAliveTime时间为0说明多余线程马上停止,队列它用的是```new LinkedBlockingQueue<Runnable>()```,这里源码点进去看发现指定队列容量为无穷大。总结的说,就是**线程池大小固定,任务队列无界**
/**
* Creates a thread pool that reuses a fixed number of threads
* operating off a shared unbounded queue. At any point, at most
* {@code nThreads} threads will be active processing tasks.
* If additional tasks are submitted when all threads are active,
* they will wait in the queue until a thread is available.
* If any thread terminates due to a failure during execution
* prior to shutdown, a new one will take its place if needed to
* execute subsequent tasks. The threads in the pool will exist
* until it is explicitly {@link ExecutorService#shutdown shutdown}.
*
* @param nThreads the number of threads in the pool
* @return the newly created thread pool
* @throws IllegalArgumentException if {@code nThreads <= 0}
*/
public static ExecutorService newFixedThreadPool(int nThreads) {
return new ThreadPoolExecutor(nThreads, nThreads,
0L, TimeUnit.MILLISECONDS,
new LinkedBlockingQueue<Runnable>());
}
|
/**
* Creates a thread pool that creates new threads as needed, but
* will reuse previously constructed threads when they are
* available. These pools will typically improve the performance
* of programs that execute many short-lived asynchronous tasks.
* Calls to {@code execute} will reuse previously constructed
* threads if available. If no existing thread is available, a new
* thread will be created and added to the pool. Threads that have
* not been used for sixty seconds are terminated and removed from
* the cache. Thus, a pool that remains idle for long enough will
* not consume any resources. Note that pools with similar
* properties but different details (for example, timeout parameters)
* may be created using {@link ThreadPoolExecutor} constructors.
*
* @return the newly created thread pool
*/
public static ExecutorService newCachedThreadPool() {
return new ThreadPoolExecutor(0, Integer.MAX_VALUE,
60L, TimeUnit.SECONDS,
new SynchronousQueue<Runnable>());
}
|
/**
* Creates an Executor that uses a single worker thread operating
* off an unbounded queue. (Note however that if this single
* thread terminates due to a failure during execution prior to
* shutdown, a new one will take its place if needed to execute
* subsequent tasks.) Tasks are guaranteed to execute
* sequentially, and no more than one task will be active at any
* given time. Unlike the otherwise equivalent
* {@code newFixedThreadPool(1)} the returned executor is
* guaranteed not to be reconfigurable to use additional threads.
*
* @return the newly created single-threaded Executor
*/
public static ExecutorService newSingleThreadExecutor() {
return new FinalizableDelegatedExecutorService
(new ThreadPoolExecutor(1, 1,
0L, TimeUnit.MILLISECONDS,
new LinkedBlockingQueue<Runnable>()));
}
`
- 应用场景:它的特性保证所有了所有的任务在一个线程中按顺序运行。所以它适用于在逻辑上需要单线程处理任务的场景。由于阻塞队列无限大,同样可能会出现FixedThreadPool的耗时过长时产生的内存问题。
2018-04-16 06:05AM
更新中…