Threads and thread pools. Why use thread pools
Threads in Android
In Android, there is a distinction between main thread and child thread. The main thread, also known as UI thread, is mainly used to deal with some things related to the interface, while the sub-thread is mainly used to deal with some time-consuming tasks, such as some network operations, IO requests and so on. If you process these time-consuming tasks in the main thread, you may experience ANR (App freezes directly).
Thread pools in Android
Thread pool, from the meaning of the name we know that thread pool is a pool containing threads, it plays a role in creating threads, managing threads, scheduling threads and so on.
3. Why use thread pools
Since threads are already a concept in Android, why use thread pools? We have two reasons for using thread pools.
- First of all, the creation and destruction of threads are performance consumption. If there are a large number of network requests in a period of time, multiple threads need to be created and destroyed, and performance loss can be imagined.
- Second, the presence of multiple threads can take up CPU execution time. We know that if there is only one CPU, the execution of a thread is all about the CPU taking turns allocating execution time to each thread. If multiple child threads exist at the same time, the corresponding CPU execution time allocated to the main thread will also be reduced, so the App is likely to stall. In view of the above two reasons, we introduced the concept of thread pool, thread creation, scheduling, destruction, etc., are managed by thread pool, so that thread reuse can be achieved, and there is no need to create a new thread every time, reducing the performance loss of thread creation and destruction. At the same time, the thread pool will limit the number of threads created to keep the number of threads in the App in a controllable range, so that multiple threads can control the resources of the main thread.
Introduction to thread pools in Android
1. Thread pool construction
In Android, thread pools are ThreadPoolExecutor objects. Let’s start by looking at the constructor of ThreadPoolExecutor.
public ThreadPoolExecutor(int corePoolSize,
int maximumPoolSize,
long keepAliveTime,
TimeUnit unit,
BlockingQueue<Runnable> workQueue,
ThreadFactory threadFactory,
RejectedExecutionHandler handler)
Copy the code
The first parameter corePoolSize is the number of core threads, which means that there are at least this many threads in the pool, even if those threads are not executing tasks. The exception is that if allowCoreThreadTimeOut is set to true in the thread pool, the core thread will also be destroyed after the keepAliveTime expires. The second parameter maximumPoolSize is the maximum number of threads in the thread pool. When the number of active threads reaches this point, subsequent new tasks are blocked. The third parameter keepAliveTime is the keepAliveTime of the thread, which means that if there are more threads in the thread pool than the number of core threads, the time starts from the moment when the thread has no tasks. If the keepAliveTime is exceeded and no new tasks come, the thread will be destroyed. Also, if allowCoreThreadTimeOut is set to true, this is the timeout specified in the first item above. The fourth parameter unit is the timing unit of the third parameter, including milliseconds, seconds, and so on. The fifth parameter, workQueue, is the queue of tasks in the thread pool that holds the Runnable object passed by the execute method (the Runnable object is a task). The type of this task queue is BlockQueue, that is, blocking queue. When the number of tasks in the queue is 0, the task fetching operation will be blocked. When the number of tasks in the queue is full (the maximum number of active threads is reached), the add operation will block. The sixth parameter, threadFactory, is a threadFactory that is used to supply a thread to the thread pool when it needs to create a new thread. The seventh argument handler is the rejection policy that the thread pool uses when the queue is bounded (that is, the fifth argument) and the task is added to the thread pool if the queue is full.
2. Classification of thread pools
I, FixedThreadPool
public static ExecutorService newFixedThreadPool(int nThreads) {
return new ThreadPoolExecutor(nThreads, nThreads,
0L, TimeUnit.MILLISECONDS,
new LinkedBlockingQueue<Runnable>());
}
Copy the code
You can see that the FixedThreadPool build calls the constructor of ThreadPoolExecutor. You can see several characteristics of FixedThreadPool from the above call:
- The number of core threads is the same as the maximum number of threads, which means that none of the threads will be destroyed while they are idle (because the maximum number of threads is the same as the number of core threads). However, the keepAliveTime is set to 0, which means that idle threads are immediately destroyed when allowCoreThreadTimeOut is set to true. When all threads are active, new tasks are in a wait state and executed when there are idle threads.
II, CacheThreadPool
public static ExecutorService newCachedThreadPool() {
return new ThreadPoolExecutor(0, Integer.MAX_VALUE,
60L, TimeUnit.SECONDS,
new SynchronousQueue<Runnable>());
}
Copy the code
You can see that the CacheThreadPool build calls the constructor of ThreadPoolExecutor. You can see several characteristics of CacheThreadPool from the above call:
- The number of core threads is zero and the maximum number of threads is integer.max_value. The keepAliveTime of a thread is 60 seconds. This means that when all threads in the thread pool are active, a new thread is created to perform the task; Idle threads are used when there are idle threads in the thread pool (there are no core threads, but threads have 60 seconds to live and are not immediately destroyed when idle). The SynchronousQueue SynchronousQueue blocks the queue. That is, the queue does not have any capacity and tasks can be added to it only when there are threads available to execute tasks. As you can see from the CacheThreadPool feature above, threads are guaranteed to execute tasks at any time (either new or idle), so there is no blocking. BlockQueue: Java Multithreading tools BlockQueue: Java Multithreading tools BlockQueue: Java Multithreading tools Therefore, CacheThreadPool can be used when a large number of tasks need to be performed but the time required to perform these tasks is relatively small.
III, ScheduledThreadPool
public static ScheduledExecutorService newScheduledThreadPool(int corePoolSize) {
return new ScheduledThreadPoolExecutor(corePoolSize);
}
public ScheduledThreadPoolExecutor(int corePoolSize) {
super(corePoolSize, Integer.MAX_VALUE, 0, NANOSECONDS,
new DelayedWorkQueue());
}
public ThreadPoolExecutor(int corePoolSize,
int maximumPoolSize,
long keepAliveTime,
TimeUnit unit,
BlockingQueue<Runnable> workQueue) {
this(corePoolSize, maximumPoolSize, keepAliveTime, unit, workQueue,
Executors.defaultThreadFactory(), defaultHandler);
}
Copy the code
Can see ScheduledThreadPoolExecutor build call ThreadPoolExecutor constructor. Can be seen from the above call ScheduledThreadPoolExecutor several features:
- The number of core threads is fixed, and the maximum number of threads is integer.max_value. The keepAliveTime of a thread is 0. This means that threads in the thread pool that exceed the number of core threads are destroyed as soon as they become idle. The ScheduledThreadPool is used to execute scheduled tasks or repetitive tasks with a fixed period. Executor.schedule (with several different overloaded methods) can be used to perform different delayed, timed, repetitive tasks.
IV, SingleThreadExecutor
public static ExecutorService newSingleThreadExecutor() {
return new FinalizableDelegatedExecutorService
(new ThreadPoolExecutor(1, 1,
0L, TimeUnit.MILLISECONDS,
new LinkedBlockingQueue<Runnable>()));
}
Copy the code
You can see that the build of SingleThreadExecutor calls the constructor of ThreadPoolExecutor. From the above call, you can see several features of SingleThreadExecutor:
- The number of core threads is 1, the maximum number of threads is also 1, and the keepAliveTime is 0, which means that only one thread can exist in the thread pool, and that thread, whether running or idle, will not be destroyed. SingleThreadExecutor ensures that all tasks are executed sequentially in a single thread.