Android's four major components Service

Android's four major components Service

Recently, I haven't found a job yet, so I'm taking advantage of the time I have to sort out the previous things, otherwise it's easy to forget. Today I'm going to explain the usage of Service. As one of the four major components of Android, its importance can be imagined. In the application, we mainly use it to perform some background operations, without interacting with the application UI, and perform time-consuming tasks.

The official documentation says:

A Service is an application component that can perform long-running operations in the background without providing a user interface. Services can be started by other application components, and they continue to run in the background even if the user switches to another application.

Additionally, components can bind to services to interact with them and even perform inter-process communication (IPC). For example, services can handle network transactions, play music, perform file I/O, and so on.

or interact with content providers, all in the background.

Service Purpose:

1. Perform time-consuming operations in the background, but do not require user interaction. 2. Some functions exposed by an application for use by other applications.

One thing that needs to be stated here is that the Service runs in the main thread, so if you need to perform time-consuming operations or access the network, you need to start another thread in the Service to execute it (if you use IntentService, you don’t need to manually start the thread yourself).

Start Service

There are two ways to start a Service:

  1. Context.startService()
  1. Context.bindService()

(Picture taken from the official document: https://developer.android.com...)

When we start a service with the startService() method, we cannot control the service after it is started, and the service will always run in the background. Even if some of the code in it is executed, if we want to terminate the service, we need to call the stopSelf() method in its code or directly call the stopService() method. When the service is started with the bindService() method, the client will obtain a persistent connection to the service, and the client will obtain an IBinder object returned by the onBind(Intent) method of the service, which is used for the client to call back the callback method in the service.

No matter which method we use, we need to define a class, let it inherit the Service class, and rewrite several methods. If we use the startService() method to start, we only need to rewrite the onCreate(), onStartCommand(Intent intent, int flags, int startId), and onDestroy() methods (in fact, we can also rewrite them). If we use the bindService() method to start, we need to rewrite the onCreate(), onBind(Intent intent), and onUnbind(Intent intent) methods. Note that as one of the four major components, Service must be configured in the manifest file before use.

  1. <application>
  2. ......
  3. <service
  4. android: name = ".MyService" >
  5. </service>
  6. </application>

Context.startService()

Code of MyService.java:

  1. public class MyService extends Service {
  2. public MyService() {
  3. }
  4.  
  5. @Override
  6. public void onCreate() {
  7. super.onCreate();
  8.  
  9. Log.i( "test" , "onCrete executed !" );
  10. }
  11.  
  12. @Override
  13. public   int onStartCommand(Intent intent, int flags, int startId) {
  14.  
  15. Log.i( "test" , "onStartComand executed !" );
  16. return super.onStartCommand(intent, flags, startId);
  17. }
  18.  
  19. @Override
  20. public void onDestroy() {
  21. super.onDestroy();
  22. Log.i( "test" , "onDestroy executed !" );
  23. }
  24. }

The code of MainActivity.java is as follows:

  1. public class MainActivity extends AppCompatActivity implements View .OnClickListener{
  2.  
  3. Button btnStart,btnStop;
  4. @Override
  5. protected void onCreate(Bundle savedInstanceState) {
  6. super.onCreate(savedInstanceState);
  7. setContentView(R.layout.activity_main);
  8.  
  9. btnStart = (Button) findViewById(R.id.btn_start);
  10. btnStop = (Button) findViewById(R.id.btn_stop);
  11. btnStart.setOnClickListener(this);
  12. btnStop.setOnClickListener(this);
  13. }
  14.  
  15.  
  16. @Override
  17. public void onClick( View   view ) {
  18.  
  19. Intent mIntent = new Intent(MainActivity.this,MyService.class);
  20.  
  21. switch ( view .getId()){
  22. case R.id.btn_start:
  23. startService(mIntent);
  24. break;
  25. case R.id.btn_stop:
  26. stopService(mIntent);
  27. break;
  28. }
  29. }
  30. }

The main interface has two buttons, one for starting the Service and one for stopping the Service:

Next, we click the START button, and the Log information is as follows:

It can be seen that the onCreate() method is executed first, and then the onStartCommand() method is executed immediately. What if we click the start button again? The result is as follows:

We can see that this time the onCreate() method is not executed again, but the onStartCommand() method is executed directly. This is because the onCreate() method is only executed when the Service is created for the first time. If it has already been created, then when startService() is called again to start the Service, only the onStartCommand() method will be executed, and the onCreate() method will not be executed again.

Next we click the stop button and you can see that the onDestroy() method is executed:

Note that if we do not click the stop button to manually stop the Service, the Service will continue to run in the background, even if the code in its onStartCommand() method has been executed, as we can see in the figure below:

At this time, our Service is always executed in the background, even if the code in its onStartCommand() method has been executed. If we want it to stop automatically, we can modify the code in the onStartCommand() method as follows:

  1. @Override
  2. public   int onStartCommand(Intent intent, int flags, int startId) {
  3.  
  4. Log.i( "test" , "onStartComand() executed !" );
  5. stopSelf();
  6. return super.onStartCommand(intent, flags, startId);
  7. }

Context.bindService()

The code for this method is slightly longer than before, because we need to control the Service on the client side, so we create an anonymous inner class ServiceConnection in MainActivity, and then pass it in the bindService() method and unbindService() method. The code in MyService.java is as follows:

  1. public class MyService extends Service {
  2. private MyBinder myBinder = new MyBinder();
  3.  
  4. public MyService() {
  5. }
  6.  
  7. @Override
  8. public void onCreate() {
  9. super.onCreate();
  10. Log.i( "test" , "onCreate() executed !" );
  11. }
  12.  
  13. @Override
  14. public void onDestroy() {
  15. super.onDestroy();
  16. Log.i( "test" , "onDestroy() executed !" );
  17. }
  18.  
  19. @Override
  20. public boolean onUnbind(Intent intent) {
  21.  
  22. Log.i( "test" , "onUnbind executed !" );
  23. return super.onUnbind(intent);
  24. }
  25.  
  26. @Override
  27. public IBinder onBind(Intent intent) {
  28. Log.i( "test" , "onBind() executed !" );
  29. return myBinder;
  30. }
  31.  
  32. class MyBinder extends Binder{
  33. public void startDownload(){
  34.  
  35. Log.i( "test" , "startDownload() in MyBinder executed!" );
  36. // To execute a specific download task, you need to start a child thread and execute specific code in it
  37. }
  38. }
  39. }

The code of MainActivity.java is as follows:

  1. public class MainActivity extends AppCompatActivity implements View .OnClickListener{
  2.  
  3. Button btnBind,btnUnBind;
  4. MyService.MyBinder myBinder;
  5. @Override
  6. protected void onCreate(Bundle savedInstanceState) {
  7. super.onCreate(savedInstanceState);
  8. setContentView(R.layout.activity_main);
  9.  
  10. btnBind = (Button) findViewById(R.id.bind);
  11. btnUnBind = (Button) findViewById(R.id.btn_unBind);
  12. btnBind.setOnClickListener(this);
  13. btnUnBind.setOnClickListener(this);
  14.  
  15. }
  16.  
  17. ServiceConnection mServiceConnection = new ServiceConnection() {
  18. @Override
  19. public void onServiceConnected(ComponentName componentName, IBinder iBinder) {
  20. // Transform IBinder down to our internal class MyBinder
  21. myBinder = (MyService.MyBinder) iBinder;
  22. // Execute the download task
  23. myBinder.startDownload();
  24. }
  25.  
  26. @Override
  27. public void onServiceDisconnected(ComponentName componentName) {
  28.  
  29. }
  30. };
  31.  
  32. @Override
  33. public void onClick( View   view ) {
  34.  
  35. Intent mIntent = new Intent(MainActivity.this,MyService.class);
  36.  
  37. switch ( view .getId()){
  38. case R.id.bind:
  39. // Bind Service
  40. bindService(mIntent,mServiceConnection,BIND_AUTO_CREATE);
  41. break;
  42. case R.id.btn_unBind:
  43. // Unbind Service
  44. unbindService(mServiceConnection);
  45. break;
  46. }
  47. }
  48. }

Click the Bind button;

Click the Unbind button:

Note that if we click Cancel Bind directly without clicking Bind first, the program will crash directly with the following error:

  1. java.lang.IllegalArgumentException: Service not registered: com.qc.admin.myserializableparceabledemo.MainActivity$1@8860e28
  2. at android.app.LoadedApk.forgetServiceDispatcher(LoadedApk.java:1120)
  3. at android.app.ContextImpl.unbindService(ContextImpl.java:1494)
  4. at android.content.ContextWrapper.unbindService(ContextWrapper.java:616)
  5. at com.qc.admin.myserializableparceabledemo.MainActivity.onClick(MainActivity.java:71)

Careful people may have already discovered that the sentence "onServiceDisconnected executed!" is not printed in the Log, which means that the onServiceDisconnected() method is not called? From a literal understanding, the onServiceConnected() method is called when the Service establishes a connection. Shouldn't onServiceDisconnected() be called when the Service disconnects? In fact, it is not. We can see it by looking at the documentation of the method:

Called when a connection to the Service has been lost. This typically happens when the process hosting the service has crashed or been killed. This does not remove the ServiceConnection itself -- this binding to the service will remain active, and you will receive a call to onServiceConnected(ComponentName, IBinder) when the Service is next running.

This means: this method will be called when the connection bound to the Service is lost, typically when the process holding the Service crashes or is killed. But this does not remove the ServiceConnection itself - it remains active, and the onServiceConnected(ComponentName, IBinder) method will still be called when the Service is executed next time.

But please note that if we follow what we just said and do not click the bindService() method first, but directly click the unbindService() method, the program will crash, but the onServiceDisconnected() method will not be called. This is easy to understand. After all, there is no connection, so how can we disconnect it? But what if we have bound the Service and then terminate the Service directly in the background? What will be the result? The answer is that the onServiceDisconnected() method will still not be called. Here I think it should be that the process ends only in unexpected circumstances, and it is called automatically by the system, rather than being stopped manually by us. We can check the comments inside this method:

This is called when the connection with the service has been

unexpectedly disconnected -- that is, its process crashed.Because it

is running in our same process, we should never see this happen.

This text clearly explains the scenario in which this method is executed: the connection is disconnected due to an exception. That is, the process crashes. Because it runs in the process where our application is located, we will never want to see this happen.

Using Context.startService() and Context.bindService() at the same time

These two methods can be used at the same time, but please note that the startService() and stopService() methods are corresponding, and the bindService() and unBind() methods are corresponding, that is, if we call startService() first and then bindService(), or vice versa, then if we only call stopService() or only call bindService(), we cannot stop the Service. We can only call them at the same time.

Let's look at the specific code:

MainActivity.java

  1. public class MainActivity extends AppCompatActivity implements View .OnClickListener{
  2.  
  3. Button btnStart,btnStop,btnBind,btnUnBind;
  4. MyService.MyBinder myBinder;
  5. @Override
  6. protected void onCreate(Bundle savedInstanceState) {
  7. super.onCreate(savedInstanceState);
  8. setContentView(R.layout.activity_main);
  9.  
  10. btnStart = (Button) findViewById(R.id.btn_start);
  11. btnStop = (Button) findViewById(R.id.btn_stop);
  12. btnBind = (Button) findViewById(R.id.btn_bind);
  13. btnUnBind = (Button) findViewById(R.id.btn_unBind);
  14.  
  15. btnStart.setOnClickListener(this);
  16. btnStop.setOnClickListener(this);
  17. btnBind.setOnClickListener(this);
  18. btnUnBind.setOnClickListener(this);
  19.  
  20. }
  21.  
  22. ServiceConnection mServiceConnection = new ServiceConnection() {
  23. @Override
  24. public void onServiceConnected(ComponentName componentName, IBinder iBinder) {
  25. // Transform IBinder down to our internal class MyBinder
  26. myBinder = (MyService.MyBinder) iBinder;
  27. // Execute the download task
  28. myBinder.startDownload();
  29.  
  30. }
  31.  
  32. @Override
  33. public void onServiceDisconnected(ComponentName componentName) {
  34.  
  35. Log.i( "test" , "onServiceDisconnected executed !" );
  36. }
  37. };
  38.  
  39. @Override
  40. public void onClick( View   view ) {
  41.  
  42. Intent mIntent = new Intent(MainActivity.this,MyService.class);
  43.  
  44. switch ( view .getId()){
  45. case R.id.btn_start:
  46. // Start the Service
  47. startService(mIntent);
  48. break;
  49. case R.id.btn_stop:
  50. //Terminate the Service
  51. stopService(mIntent);
  52. break;
  53. case R.id.btn_bind:
  54. // Bind Service
  55. bindService(mIntent,mServiceConnection,BIND_AUTO_CREATE);
  56. break;
  57. case R.id.btn_unBind:
  58. // Unbind Service
  59. unbindService(mServiceConnection);
  60. break;
  61. }
  62. }
  63. }

Code of MyService.java:

  1. public class MyService extends Service {
  2. private MyBinder myBinder = new MyBinder();
  3.  
  4. public MyService() {
  5. }
  6.  
  7. @Override
  8. public void onCreate() {
  9. super.onCreate();
  10. Log.i( "test" , "onCreate() executed !" );
  11. }
  12.  
  13. @Override
  14. public   int onStartCommand(Intent intent, int flags, int startId) {
  15.  
  16. Log.i( "test" , "onStartComand() executed !" );
  17. return super.onStartCommand(intent, flags, startId);
  18. }
  19.  
  20. @Override
  21. public void onDestroy() {
  22. super.onDestroy();
  23. Log.i( "test" , "onDestroy() executed !" );
  24. }
  25.  
  26. @Override
  27. public boolean onUnbind(Intent intent) {
  28.  
  29. Log.i( "test" , "onUnbind executed !" );
  30. return super.onUnbind(intent);
  31. }
  32.  
  33. @Override
  34. public IBinder onBind(Intent intent) {
  35. Log.i( "test" , "onBind() executed !" );
  36. return myBinder;
  37. }
  38.  
  39. class MyBinder extends Binder{
  40. public void startDownload(){
  41.  
  42. Log.i( "test" , "startDownload() in MyBinder executed!" );
  43. // Execute specific download tasks
  44. }
  45. }
  46. }

a. Below is the output of clicking the start, bind, stop, and unBind buttons in sequence:

b. Below is the output when clicking the start, bind, unbind, and stop buttons in sequence:

Running a service in the foreground

We have been saying above that Service is generally used to perform time-consuming operations in the background, but you should know that Service can also run in the foreground. Background Service has a lower priority and is likely to be killed by the system in situations such as insufficient memory. By setting it to the foreground, the chance of it being killed can be greatly reduced. The foreground Service will display an icon in the system notification bar, where we can perform some operations. Common scenarios for foreground Service include music players and weather forecasts:

Then let's go directly to the code:

  1. @Override
  2. public void onCreate() {
  3. super.onCreate();
  4. Log.i( "test" , "onCreate() executed !" );
  5.  
  6. Intent mIntent = new Intent(this, SecondActivity.class);
  7. PendingIntent mPendingIntent = PendingIntent.getActivity(this, 0, mIntent, 0);
  8. Notification mNotification = new NotificationCompat.Builder(this)
  9. .setSmallIcon(R.mipmap.ic_launcher)
  10. .setContentTitle( "My Notification " )
  11. .setContentText( "Hello World! " )
  12. .setContentIntent(mPendingIntent)
  13. .build();
  14.  
  15. // NOTE: The integer ID provided to startForeground() must not be 0.
  16. // To remove the service from the foreground, call stopForeground(). This method takes a boolean indicating whether to also remove the status bar notification.
  17. // However, stopForeground() does not stop the service. However, if you stop the service while it is running in the foreground, the notification will also be removed.
  18. startForeground(1, mNotification);
  19. }

In fact, the implementation here is very simple, that is, passing a Notification through startForeground(1, mNotification);, so as to establish an association between the Notification and the Service. When we click on this notification, it will jump to the second Activity (but the Notification will not disappear), as shown in the screenshot below:

<<:  Android touch events (notes)

>>:  Android View Focus Summary

Recommend

A nanny-level guide for beginners to create Tik Tok videos (Part 1)

Today, I will introduce to you some tips on how t...

Inventory | How did the “World Cup Game” trigger user growth?

This article is about the World Cup held every fo...

"Super Three Kingdoms Season 7" Baidu Cloud Resource Download Link Complete

One of the four great traditional classics A new ...

How to use TypeScript to improve JavaScript programming

TypeScript can help you write better JavaScript c...

Tutorial on using TensorFlow on iOS (Part 1)

Before using deep learning networks for predictiv...

19 pictures to understand advertising brand insights

Brand solves the problem of cognition. All work o...

Free is the best activity to attract new customers strategy

The event will start from August 8th and last unt...

【Dream Complete User Manual】Control your dreams and turn your life around

【Dream Complete User Manual】Control your dreams a...

Short video operation: How to achieve 1.6 million+ fans?

In short video operations , if you want to do a g...

Keep Competitive Product Analysis

With the continuous development of China's ec...