Android custom thread pool programming practice

Android custom thread pool programming practice

1. Introduction to Executor

After Java 5, concurrent programming introduced a bunch of new APIs for starting, scheduling, and managing threads. The Executor framework was introduced in Java 5, which uses a thread pool mechanism internally. It is under the java.util.cocurrent package. Through this framework, you can control the startup, execution, and shutdown of threads, which can simplify concurrent programming operations. Therefore, after Java 5, starting threads through Executor is better than using the start method of Thread. In addition to being easier to manage and more efficient (implemented with a thread pool to save overhead), there is another key point: it helps avoid the this escape problem - if we start a thread in the constructor, because another task may start executing before the constructor ends, it may access a half-initialized object using Executor in the constructor.

The Executor framework includes: thread pool, Executor, Executors, ExecutorService, CompletionService, Future, Callable, etc.

In the Java code, Executor is an interface with only one method.

  1. public interface Executor {
  2.   
  3. /**
  4. * Executes the given command at   some   time   in the future. The command
  5. * may execute   in a new thread, in a pooled thread, or   in the calling
  6. * thread, at the discretion of the {@code Executor} implementation.
  7. *
  8. * @param command the runnable task
  9. * @throws RejectedExecutionException if this task cannot be
  10. * accepted for execution
  11. * @throws NullPointerException if command is   null  
  12. */
  13. void execute (Runnable command);
  14. }

2. ExecutorService

ExecutorService is an interface that inherits Executor. In addition to the execute(Runnable command) method, it also extends other methods:

  1. public interface ExecutorService extends Executor {
  2.  
  3. }
  • void shutdown();
  • List<Runnable> shutdownNow();
  • boolean isShutdown();
  • boolean isTerminated();
  • boolean awaitTermination(long timeout, TimeUnit unit) throws InterruptedException;
  • <T> Future<T> submit(Callable<T> task); //Submit a task
  • <T> Future<T> submit(Runnable task, T result); //Submit a task
  • Future<?> submit(Runnable task); //Submit a task
  • <T> List<Future<T>> invokeAll(Collection<? extends Callable<T>> tasks) throws InterruptedException;
  • <T> List<Future<T>> invokeAll(Collection<? extends Callable<T>> tasks, long timeout, TimeUnit unit) throws InterruptedException;
  • <T> T invokeAny(Collection<? extends Callable<T>> tasks) throws InterruptedException, ExecutionException;
  • <T> T invokeAny(Collection<? extends Callable<T>> tasks, long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException;

2.1 execute(Runnable)

Receives a java.lang.Runnable object as a parameter and executes it asynchronously. The following is an example of using ExecutorService to execute Runnable

  1. package com.app;
  2.  
  3. import java.util.concurrent.ExecutorService;
  4.  
  5. import java.util.concurrent.Executors;
  6.  
  7. public class ExecutorTest {
  8.  
  9. public   static void main(String[] args) {
  10.  
  11. //Create a thread pool with a fixed size of 10 threads
  12.  
  13. ExecutorService executorService = Executors.newFixedThreadPool( 10 );
  14.  
  15. //Execute a task. The task is a new Runnable() object.
  16.  
  17. executorService.execute (new Runnable() {
  18.  
  19. @Override
  20.  
  21. public void run() {
  22.  
  23. Log.d( Thread.currentThread().getName() );
  24.  
  25. }
  26.  
  27. });
  28.  
  29. //Close the thread pool
  30.  
  31. executorService.shutdown();
  32.  
  33. }
  34.  
  35. }

result:

  1. pool-1-thread-1

There is no way to get the result after executing Runnable using this method. If you want to get the return value after running, you must use the execute() method that receives the Callable parameter, which will be mentioned below.

2.2、submit(Runnable)

The method submit(Runnable) also receives a Runnable implementation as a parameter, but returns a Future object. This Future object can be used to determine whether the Runnable has finished executing. The following is an example of the submit() method of ExecutorService:

  1. package com.app;
  2.  
  3. import java.util.concurrent.ExecutorService;
  4.  
  5. import java.util.concurrent.Executors;
  6.  
  7. import java.util.concurrent.Future;
  8.  
  9. public class ExecutorTest {
  10.  
  11. public   static void main(String[] args) {
  12.  
  13. //Create a thread pool with a fixed size of 10 threads
  14.  
  15. ExecutorService executorService = Executors.newFixedThreadPool( 10 );
  16.  
  17. //Execute a task. The task is a new Runnable() object.
  18.  
  19. Future future = executorService.submit( new Runnable() {
  20.  
  21. @Override
  22.  
  23. public void run() {
  24.  
  25. Log.d( Thread.currentThread().getName() );
  26.  
  27. }
  28.  
  29. });
  30.  
  31. try {
  32.  
  33. // If the task is finished executing, return null  
  34.  
  35. Log.d( "" + future.get() );
  36.  
  37. } catch (Exception e) {
  38.  
  39. e.printStackTrace();
  40.  
  41. }
  42.  
  43. //Close the thread pool
  44.  
  45. executorService.shutdown();
  46.  
  47. }
  48.  
  49. }

result:

  1. pool-1-thread-1
  2.  
  3. null  

2.3 submit(Callable)

The methods submit(Callable) and submit(Runnable) are similar, but the difference is that they accept different parameter types. Callable instances are very similar to Runnable instances, but the call() method of Callable can return a result. The method Runnable.run() cannot return a result.

The return value of Callable can be obtained from the Future object returned by the submit(Callable) method. The following is an example of ExecutorService Callable:

  1. package com.app;
  2.  
  3. import java.util.concurrent.Callable;
  4.  
  5. import java.util.concurrent.ExecutorService;
  6.  
  7. import java.util.concurrent.Executors;
  8.  
  9. import java.util.concurrent.Future;
  10.  
  11. public class ExecutorTest {
  12.  
  13. public   static void main(String[] args) {
  14.  
  15. //Create a thread pool with a fixed size of 10 threads
  16.  
  17. ExecutorService executorService = Executors.newFixedThreadPool( 10 );
  18.  
  19. //Execute a task. The task is a new Callable() object.
  20.  
  21. Future future = executorService.submit( new Callable<String>() {
  22.  
  23. @Override
  24.  
  25. public String call() throws Exception {
  26.  
  27. return   "Execution completed" ;
  28.  
  29. }
  30.  
  31. }) ;
  32.  
  33. try {
  34.  
  35. // If the task is finished executing, return
  36.  
  37. Log.d( "The result is: " + future.get());
  38.  
  39. } catch (Exception e) {
  40.  
  41. e.printStackTrace();
  42.  
  43. }
  44.  
  45. //Close the thread pool
  46.  
  47. executorService.shutdown();
  48.  
  49. }
  50.  
  51. }

result:

The result is: Execution completed

2.4、inVokeAny()

The invokeAny() method receives a collection of Callable objects as a parameter. Calling this method does not return a Future object, but returns the result of a Callable object in the collection. There is no guarantee which Callable the result will be after the call, but only that it is a Callable object that has finished executing. If a task is completed or an exception is thrown, the method will cancel the execution of other Callables.

  1. package com.app;
  2.  
  3. import java.util.ArrayList;
  4.  
  5. import java.util.List;
  6.  
  7. import java.util.concurrent.Callable;
  8.  
  9. import java.util.concurrent.ExecutionException;
  10.  
  11. import java.util.concurrent.ExecutorService;
  12.  
  13. import java.util.concurrent.Executors;
  14.  
  15. public class ExecutorTest {
  16.  
  17. public   static void main(String[] args) {
  18.  
  19. //Create a thread pool with a fixed size of 10 threads
  20.  
  21. ExecutorService executorService = Executors.newFixedThreadPool( 10 );
  22.  
  23. List<Callable<String>> list = new ArrayList<>();
  24.  
  25. //Create *** Callable
  26.  
  27. Callable<String> callable1 = new Callable<String>() {
  28.  
  29. @Override
  30.  
  31. public String call() throws Exception {
  32.  
  33. Log.d( "callable 1 thread is: " + Thread.currentThread().getName() );
  34.  
  35. return   "Completed executing callable 1" ;
  36.  
  37. }
  38.  
  39. };
  40.  
  41. //Create the second Callable
  42.  
  43. Callable<String> callable2 = new Callable<String>() {
  44.  
  45. @Override
  46.  
  47. public String call() throws Exception {
  48.  
  49. Log.d( "callable 2 thread is: " + Thread.currentThread().getName() );
  50.  
  51. return   "Completed executing callable 2" ;
  52.  
  53. }
  54.  
  55. };
  56.  
  57. list. add ( callable1 ) ;
  58.  
  59. list. add ( callable2 ) ;
  60.  
  61. try {
  62.  
  63. String result = executorService.invokeAny( list );
  64.  
  65. Log.d( "The result is: " + result);
  66.  
  67. } catch (InterruptedException e1) {
  68.  
  69. e1.printStackTrace();
  70.  
  71. } catch (ExecutionException e1) {
  72.  
  73. e1.printStackTrace();
  74.  
  75. }
  76.  
  77. //Close the thread pool
  78.  
  79. executorService.shutdown();
  80.  
  81. }
  82.  
  83. }

result:

The callable 1 thread is: pool-1-thread-1

The callable 2 thread is: pool-1-thread-2

The result is: Callable 2 has been executed

Summarize:

1. You can see that the call methods in Callable are all run in the child thread.

2. executorService.invokeAny( list ); The return value is the return value of any Callable. Any one is possible.

2.5, invokeAll()

The method invokeAll() calls all Callable objects in the parameter collection and returns a collection of Future objects. You can use this returned collection to manage the execution results of each Callable. It should be noted that a task may end due to an exception, so it may not really run successfully. But we have no way to understand this difference through Future objects.

  1. package com.app;
  2.  
  3. import java.util.ArrayList;
  4.  
  5. import java.util.List;
  6.  
  7. import java.util.concurrent.Callable;
  8.  
  9. import java.util.concurrent.ExecutorService;
  10.  
  11. import java.util.concurrent.Executors;
  12.  
  13. import java.util.concurrent.Future;
  14.  
  15. public class ExecutorTest {
  16.  
  17. public   static void main(String[] args) {
  18.  
  19. //Create a thread pool with a fixed size of 10 threads
  20.  
  21. ExecutorService executorService = Executors.newFixedThreadPool( 10 );
  22.  
  23. List<Callable<String>> list = new ArrayList<>();
  24.  
  25. //Create *** Callable
  26.  
  27. Callable<String> callable1 = new Callable<String>() {
  28.  
  29. @Override
  30.  
  31. public String call() throws Exception {
  32.  
  33. Log.d( "callable 1 thread is: " + Thread.currentThread().getName() );
  34.  
  35. return   "Completed executing callable 1" ;
  36.  
  37. }
  38.  
  39. };
  40.  
  41. //Create the second Callable
  42.  
  43. Callable<String> callable2 = new Callable<String>() {
  44.  
  45. @Override
  46.  
  47. public String call() throws Exception {
  48.  
  49. Log.d( "callable 2 thread is: " + Thread.currentThread().getName() );
  50.  
  51. return   "Completed executing callable 2" ;
  52.  
  53. }
  54.  
  55. };
  56.  
  57. list. add ( callable1 ) ;
  58.  
  59. list. add ( callable2 ) ;
  60.  
  61. List<Future<String>> result;
  62.  
  63. try {
  64.  
  65. result = executorService.invokeAll(list);
  66.  
  67. for (Future<String> future : result) {
  68.  
  69. Log.d( "The result is: " + future.get());
  70.  
  71. }
  72.  
  73. } catch (Exception e) {
  74.  
  75. e.printStackTrace();
  76.  
  77. }
  78.  
  79. //Close the thread pool
  80.  
  81. executorService.shutdown();
  82.  
  83. }
  84.  
  85. }

result

The callable 1 thread is: pool-1-thread-1

The callable 2 thread is: pool-1-thread-2

The result is: Callable 1 has been executed

The result is: Callable 2 has been executed

Note: 1: The call method of Callable is executed in the child thread

2: executorService.invokeAll( list ) is the return value. However, it will only be returned after all Callable objects have been executed. The return value is a list, and the order is the same as List<Callable>. During the execution process, if any Callable has an exception, the program will crash and no value will be returned.

2.6 How to shut down the ExecuteService service?

When you are done using the ExecutorService, you should close it to ensure that the threads do not continue to run. For example, if your program is started through the main() method and the main thread exits your program, if you still have an active ExecutorService in your program, then the program will continue to run. Active threads in the ExecutorService will prevent the Java virtual machine from shutting down.

To shut down the threads in the ExecutorService, you need to call the shutdown() method. The ExecutorService does not shut down immediately, but stops accepting new tasks. Once all threads finish executing the current tasks, the ExecutorServie will be truly shut down. All tasks submitted to the ExecutorService before calling the shutdown() method will be executed.

If you want to shut down the ExecutorService immediately, you can call shutdownNow(). This method will try to shut down all running tasks immediately and skip all tasks that have been submitted but not yet run. However, there is no guarantee that the running tasks can be successfully shut down. It is possible that they will be shut down, or they may continue to run until the task is completed. This is a last resort attempt.

<<:  Lighthouse permissions are open, is open source technology the future of VR?

>>:  What will the VR 2.0 era be like? Five major changes may be coming

Recommend

Apple releases iOS 12 GM version: officially pushed on September 18

At Apple's Gather Round conference early this...

Tik Tok data-based operation, promotion and analysis skills!

Today I will give you an in-depth analysis of Dou...

How to design a live broadcast room? Live broadcast room design guide!

Live streaming has long been the fastest and most...

Is it safe for everyone to take a hot spring bath in the same pool in winter?

Produced by: Science Popularization China Author:...

How to build an excellent entrepreneurial team? How to build a SEO team?

A professional SEO team is crucial to the operati...

What are the differences between 400 telephones and ordinary telephones?

Before the emergence of 400 telephone numbers , t...

Can we still happily use public free Wi-Fi?

Recently, I have seen many reports saying that fr...