The most elegant way to exit an Android app

The most elegant way to exit an Android app

[[185951]]

Let's first look at some common exit methods (inelegant ways)

1. Container Type

Create a global container to store all Activities, and loop through all Activities when exiting

  1. import java.util.ArrayList;
  2. import java.util.List;
  3. import android.app.Activity;
  4. import android.os.Bundle;
  5. public class BaseActivity extends Activity {
  6. @Override
  7. protected void onCreate(Bundle savedInstanceState) {
  8. super.onCreate(savedInstanceState);
  9. // Add Activity to the stack
  10. AtyContainer.getInstance().addActivity(this);
  11. }
  12. @Override
  13. protected void onDestroy() {
  14. super.onDestroy();
  15. // End the Activity & remove it from the stack
  16. AtyContainer.getInstance().removeActivity(this);
  17. }
  18. }
  19. class AtyContainer {
  20. private AtyContainer() {
  21. }
  22. private static AtyContainer instance = new AtyContainer();
  23. private static List activityStack = new ArrayList();
  24. public   static AtyContainer getInstance() {
  25. return instance;
  26. }
  27. public void addActivity(Activity aty) {
  28. activityStack.add (aty) ;
  29. }
  30. public void removeActivity(Activity aty) {
  31. activityStack.remove(aty);
  32. }
  33. /**
  34. * End all Activities
  35. */
  36. public void finishAllActivity() {
  37. for ( int i = 0, size = activityStack. size (); i if ( null != activityStack.get(i)) {
  38. activityStack.get(i).finish();
  39. }
  40. }
  41. activityStack.clear();
  42. }
  43. }

This method is relatively simple, but you can see that activityStack holds a strong reference to the Activity, which means that when an Activity exits abnormally, activityStack does not release the reference immediately, which will cause memory problems. Next, let's look at a similar method, but it will be a little more elegant.

2. Broadcast

By registering a broadcast in BaseActivity, a broadcast is sent when exiting, and finish exiting

  1. public class BaseActivity extends Activity {
  2. private static final String EXITACTION = "action.exit" ;
  3. private ExitReceiver exitReceiver = new ExitReceiver();
  4. @Override
  5. protected void onCreate(Bundle savedInstanceState) {
  6. super.onCreate(savedInstanceState);
  7. IntentFilter filter = new IntentFilter();
  8. filter.addAction(EXITACTION);
  9. registerReceiver(exitReceiver, filter);
  10. }
  11. @Override
  12. protected void onDestroy() {
  13. super.onDestroy();
  14. unregisterReceiver(exitReceiver);
  15. }
  16. class ExitReceiver extends BroadcastReceiver {
  17. @Override
  18. public void onReceive(Context context, Intent intent) {
  19. BaseActivity.this.finish();
  20. }
  21. }
  22. }

3. Process

End the application by directly killing the current application process. It is simple, crude, and effective!

  1. android.os.Process.killProcess(android.os.Process.myPid());
  2. System.exit(0);
  3. ActivityManager manager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
  4. manager.killBackgroundProcesses(getPackageName());

All three methods can achieve the same effect, but on the simulator, a message box "Unfortunately, XXX has stopped" will pop up, but the app can indeed be exited. Some real devices will fail directly and can only finish the current Activity (such as the Xiaomi Note in my hand. Several domestic ROM fw layers have too many changes, so use this method with caution)

4. RS Elegant Style

What is RS style? It is Receiver+singleTask. We know that there are four loading modes for Activity, and singleTask is one of them. After using this mode, when starting Activity, it will first query whether there is an instance of Activity in the current stack. If it exists, it will be placed on the top of the stack and all the Activities above it will be removed from the stack. When we open an app, the first thing is a splash page, and then the splash page will be finished. Jump to the home page. Then it will jump to the home page N times, during which an indefinite number of Activities will be generated, some of which will be destroyed, and some will reside in the stack, but the bottom of the stack will always be our HomeActivity. This makes the problem much simpler. We only need two steps to exit the app elegantly.

1. Register an exit broadcast in HomeActivity, which is the same as the second broadcast, but here you only need to register on one page of HomeActivity.

2. Set the startup mode of HomeActivity to singleTask.

When we need to exit, we only need to startActivity(this,HomeActivity,class), and then send an exit broadcast. The above code will first remove all Activities above HomeActivity from the stack, and then receive the broadcast and finish itself. Everything is OK! No pop-up window, no need to consider model Rom adaptation. There will be no memory problem, it is so elegant and simple!

5. SingleTask Redesign

After communicating with some guys, many of them said that registering for broadcasting was a bit troublesome. The guy downstairs proposed a simpler way, and the idea was also very simple.

1. Set the loading mode of MainActivity to singleTask

2. Rewrite the onNewIntent method in MainActivity

3. Add an exit tag to the Intent when you need to exit

Since many friends are eager for source code, we will explain this method directly in the form of code.

The first step is to set the loading mode of MainActivity to singleTask

  1. android:launchMode= "singleTask"  

The second step is to rewrite the onNewIntent() method

  1. private static final String TAG_EXIT = "exit" ;
  2. @Override
  3. protected void onNewIntent(Intent intent) {
  4. super.onNewIntent(intent);
  5. if (intent != null ) {
  6. boolean isExit = intent.getBooleanExtra(TAG_EXIT, false );
  7. if (isExit) {
  8. this.finish();
  9. }
  10. }
  11. }

Step 3 Exit

  1. Intent intent = new Intent(this,MainActivity.class);
  2. intent.putExtra(MainActivity.TAG_EXIT, true );
  3. startActivity(intent);

6. Lazy Style

This method is simpler and only requires the following two steps

1. Set MainActivity as singleTask

2. Place the exit in MainActivity

We can see that many applications double-click the home button twice to exit the application, which is implemented in this way. Here is the source code for how to handle two consecutive clicks to exit private boolean mIsExit;

  1. @Override
  2. /**
  3. * Double-click the return key to exit
  4. */
  5. public boolean onKeyDown( int keyCode, KeyEvent event) {
  6. if (keyCode == KeyEvent.KEYCODE_BACK) {
  7. if (mIsExit) {
  8. this.finish();
  9. } else {
  10. Toast.makeText(this, "Press again to exit" , Toast.LENGTH_SHORT).show();
  11. mIsExit = true ;
  12. new Handler().postDelayed(new Runnable() {
  13. @Override
  14. public void run() {
  15. mIsExit = false ;
  16. }
  17. }, 2000);
  18. }
  19. return   true ;
  20. }
  21. return super.onKeyDown(keyCode, event);
  22. }

<<:  Withstand 10 billion requests! Make a "sure" red envelope system

>>:  Tutorial on using TensorFlow on iOS (Part 1)

Recommend

A brief introduction to KCP protocol

Part 01 What is KCP Protocol KCP is an open sourc...

Precision Telemarketing-Low-cost Telemarketing Team Development Method

Precision Telemarketing - Low-cost Telemarketing ...

Xiaohongshu "Product Operation" Analysis: How to guide new users to register?

Today’s article is the first in a series of artic...

How do you plan a successful event and ensure that expectations are met?

A good product may not be known to anyone. After ...

Kuaishou brand self-broadcast transaction formula

It is no longer a new thing for brands to sell pr...

How much does it cost to produce the Tianshui Printing mini program?

In order to better penetrate into various industr...

SEM promotion: 10 ways to find long-tail words!

I believe that most SEMers are familiar with the ...

Analysis of Toutiao's competitive products

With the rapid development of the Internet, peopl...