Android-返回桌面?退出程序?

android

有時(shí)候我們希望用戶雙擊返回鍵的時(shí)候,再去退出應(yīng)用,那么怎么實(shí)現(xiàn)這種需求呢?

有時(shí)候我們希望用戶點(diǎn)擊一次返回鍵的時(shí)候,不是退出應(yīng)用,而是返回到桌面,那么怎么實(shí)現(xiàn)這種效果呢?

如何管理整個(gè)應(yīng)用的各個(gè)環(huán)節(jié),做到退出應(yīng)用呢?

在這里,對(duì)相關(guān)方法大概整理。

一、容器式

我們?cè)贐aseActivity 建立一個(gè)全局容器,把所有的Activity存儲(chǔ)起來(lái)統(tǒng)一維護(hù),另外可以在Base里可以處理一些界面的共同邏輯,退出時(shí)循環(huán)遍歷finish所有Activity

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        // 添加Activity到堆棧
        BaseAtyContainer.getInstance().addActivity(this);
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        // 結(jié)束Activity&從棧中移除該Activity
        BaseAtyContainer.getInstance().removeActivity(this);
    }

}

class BaseAtyContainer {
    private BaseAtyContainer() {
    }

    private static BaseAtyContainer instance = new BaseAtyContainer();
    private static List<Activity> activityStack = new ArrayList<Activity>();

    public static BaseAtyContainer getInstance() {
        return instance;
    }

    public void addActivity(Activity aty) {
        activityStack.add(aty);
    }

    public void removeActivity(Activity aty) {
        activityStack.remove(aty);
    }

    /**
     * 結(jié)束所有Activity
     */
    public void finishAllActivity() {
        for (int i = 0, size = activityStack.size(); i < size; i++) {
            if (null != activityStack.get(i)) {
                activityStack.get(i).finish();
            }
        }
        activityStack.clear();
    }

} 

需要退出的界面中僅需調(diào)用以下代碼,即可完成退出應(yīng)用的邏輯。
(此方法較為簡(jiǎn)單, 不過(guò)可以發(fā)現(xiàn)activityStack持有這些Activity的強(qiáng)引用,也就是說(shuō)當(dāng)某個(gè)Activity異常退出時(shí),activityStack沒(méi)有及時(shí)釋放掉引用,可能存在導(dǎo)致內(nèi)存問(wèn)題的隱患。)

BaseAtyContainer.getInstance().finishAllActivity();

二、廣播式

通過(guò)在BaseActivity中注冊(cè)一個(gè)廣播,當(dāng)退出時(shí)發(fā)送一個(gè)廣播,finish退出

public class BaseActivity extends Activity {

    //退出的廣播頻段
    private static final String EXITACTION = "action2exit";

    private ExitReceiver exitReceiver = new ExitReceiver();

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        //界面創(chuàng)建時(shí)注冊(cè)廣播
        IntentFilter filter = new IntentFilter();
        filter.addAction(EXITACTION);
        registerReceiver(exitReceiver, filter);
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        //界面銷毀時(shí)解除廣播
        unregisterReceiver(exitReceiver);
    }

    class ExitReceiver extends BroadcastReceiver {
        @Override
        public void onReceive(Context context, Intent intent) {
            //收到廣播時(shí),finish
            BaseActivity.this.finish();
        }
    }

}

需要退出的界面中僅需調(diào)用以下代碼,即可完成退出應(yīng)用的邏輯。

   sendBroadcast(new Intent("action2exit"));

三、進(jìn)程式

通過(guò)直接殺死當(dāng)前應(yīng)用的進(jìn)程來(lái)結(jié)束應(yīng)用,簡(jiǎn)單粗暴!(不推薦!不推薦!不推薦!?。?br> 方式一:

   android.os.Process.killProcess(android.os.Process.myPid());

方式二:

   System.exit(0);

方式三(需要權(quán)限: requires android.permission.KILL_BACKGROUND_PROCESSES):

    ActivityManager manager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
    manager.killBackgroundProcesses(getPackageName());

這三種都能達(dá)到退出應(yīng)用的效果。但是部分真機(jī)會(huì)直接失效,或者只能finish當(dāng)前Activity,國(guó)產(chǎn)ROM改動(dòng)太多,不建議使用相關(guān)方法。

四、 Receiver+singleTask

我們知道Activity有四種加載模式,而singleTask就是其中的一種,使用這個(gè)模式之后,當(dāng)startActivity時(shí),它先會(huì)在當(dāng)前棧中查詢是否存在Activity的實(shí)例,如果存在,則將其至于棧頂,并將其之上的所有Activity移除棧。

那么我們打開(kāi)一個(gè)app,首先是一個(gè)splash頁(yè)面,然后會(huì)finish掉splash頁(yè)面。跳轉(zhuǎn)到主頁(yè)。然后會(huì)在主頁(yè)進(jìn)行N次的跳轉(zhuǎn),期間會(huì)產(chǎn)生數(shù)量不定的Activity,有的被銷毀,有的駐留在棧中,但是棧底永遠(yuǎn)是我們的MainActivity。這樣就讓問(wèn)題變得簡(jiǎn)單很多了。我們只需兩步操作即可優(yōu)雅的實(shí)現(xiàn)app的退出。

步驟一:
AndroidManifest.xml中找到我們的主界面Activity,并對(duì)其設(shè)置android:launchMode="singleTask"(這里以MainActivity為例)。

      <activity
            android:name=".MainActivity"
            android:launchMode="singleTask">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

步驟二:
在MainActivity注冊(cè)一個(gè)退出廣播,和第二個(gè)廣播式一樣,但是這里只需要在MainActivity一個(gè)頁(yè)面注冊(cè)即可。

        public class MainActivity extends Activity {
            //退出的廣播頻段
            private static final String EXITACTION = "action2exit";
        
            private ExitReceiver exitReceiver = new ExitReceiver();
        
            class ExitReceiver extends BroadcastReceiver {
                @Override
                public void onReceive(Context context, Intent intent) {
                    //收到廣播時(shí),finish
                    finish();
                }
            }
        
            @Override
            protected void onCreate(Bundle savedInstanceState) {
                super.onCreate(savedInstanceState);
                setContentView(R.layout.activity_main);
        
                //界面創(chuàng)建時(shí)注冊(cè)廣播
                IntentFilter filter = new IntentFilter();
                filter.addAction(EXITACTION);
                registerReceiver(exitReceiver, filter);
        
            }
        
            @Override
            protected void onDestroy() {
                super.onDestroy();
                //界面銷毀時(shí)解除廣播
                unregisterReceiver(exitReceiver);
            }
        }

至此,當(dāng)我們需要退出的時(shí)候只需要startActivity(context,MainActivity,class), 再發(fā)送一個(gè)退出廣播。上面代碼首先會(huì)把棧中MainActivity之上的所有Activity移除出棧,然后接到廣播finish()自己。一切OK ! 沒(méi)有彈框,不用考慮機(jī)型Rom適配。不會(huì)有內(nèi)存問(wèn)題,就是那么的優(yōu)雅,簡(jiǎn)單!

       startActivity(new Intent(context,MainActivity.class));
       sendBroadcast(new Intent(EXITACTION));

五、singleTask(Receiver+singleTask優(yōu)化版)

按照第四種方式,注冊(cè)廣播略顯麻煩和臃腫,這里提出一種更簡(jiǎn)單的方式,思路也很簡(jiǎn)單。
步驟一:
既然是使用singleTask處理,那么同樣需要在AndroidManifest.xml中找到我們的主界面Activity,并對(duì)其設(shè)置android:launchMode="singleTask"(這里以MainActivity為例)。

      <activity
            android:name=".MainActivity"
            android:launchMode="singleTask">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

步驟二:
MainActivity中重寫(xiě)onNewIntent()方法

       public class MainActivity extends Activity {

           private static final String TAG_EXIT = "tag2exit";

           @Override
           protected void onNewIntent(Intent intent) {
               super.onNewIntent(intent);
               if (intent != null) {
                   boolean isExit = intent.getBooleanExtra(TAG_EXIT, false);
                   if (isExit) {
                       this.finish();
                   }
               }
           }

           @Override
           protected void onCreate(Bundle savedInstanceState) {
               super.onCreate(savedInstanceState);
               setContentView(R.layout.activity_main);
           }

       }

至此,當(dāng)我們需要退出的時(shí)候只需要翼指定的tag值去startActivity即可完成退出邏輯

     Intent intent = new Intent(context,MainActivity.class);
             intent.putExtra(MainActivity.TAG_EXIT, true);
             startActivity(intent);

六、 雙擊退出(我最喜歡這種)

我們可以看到很多應(yīng)用都是雙擊兩次BACK鍵退出應(yīng)用,大部分就是基于這樣的方式來(lái)實(shí)現(xiàn)的。
很簡(jiǎn)單,
步驟一:
首先依然是使用singleTask處理,那么同樣需要在AndroidManifest.xml中找到我們的主界面Activity,并對(duì)其設(shè)置android:launchMode="singleTask"(這里以MainActivity為例)。

      <activity
            android:name=".MainActivity"
            android:launchMode="singleTask">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

2、將退出出口放置在MainActivity

      public class MainActivity extends Activity {

          @Override
          protected void onCreate(Bundle savedInstanceState) {
              super.onCreate(savedInstanceState);
              setContentView(R.layout.activity_main);
          }

          private boolean mIsExit;

          /**
           * 雙擊返回鍵退出
           */
          @Override
          public boolean onKeyDown(int keyCode, KeyEvent event) {
              if (keyCode == KeyEvent.KEYCODE_BACK) {
                  if (mIsExit) {
                      this.finish();

                  } else {
                      Toast.makeText(this, "再按一次退出", Toast.LENGTH_SHORT).show();
                      mIsExit = true;
                      new Handler().postDelayed(new Runnable() {
                          @Override
                          public void run() {
                              mIsExit = false;
                          }
                      }, 2000);
                  }
                  return true;
              }
              return super.onKeyDown(keyCode, event);
          }
      }

七、 返回桌面,不退出應(yīng)用

有時(shí)候我們的應(yīng)用我們希望他點(diǎn)擊一次返回鍵的時(shí)候不是退出應(yīng)用而是返回到桌面,比如像QQ和微信等聊天軟件,在主界面點(diǎn)擊返回鍵的時(shí)候不是退出應(yīng)用而是返回到桌面,那么怎么能夠?qū)崿F(xiàn)這種需求呢?
其實(shí)也很簡(jiǎn)單,依然只需要重寫(xiě)onKeyDown()這個(gè)方法,在方法里面實(shí)現(xiàn)返回的代碼,這樣就可以實(shí)現(xiàn)這個(gè)需求了。
具體代碼如下所示:

      public class MainActivity extends Activity {

          @Override
          protected void onCreate(Bundle savedInstanceState) {
              super.onCreate(savedInstanceState);
              setContentView(R.layout.activity_main);
          }

          @Override
          public boolean onKeyDown(int keyCode, KeyEvent event) {

              if (keyCode == KeyEvent.KEYCODE_BACK) {
                  Intent home = new Intent(Intent.ACTION_MAIN);
                  home.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
                  home.addCategory(Intent.CATEGORY_HOME);
                  startActivity(home);
                  return true;
              }
              return super.onKeyDown(keyCode, event);
          }
      }

以上,整體概括了一部分返回桌面,退出應(yīng)用的相關(guān)方法和邏輯

當(dāng)然一定還有更多方法和更優(yōu)化的邏輯,還請(qǐng)大家提出,共同完善,
有了需求才有了功能,有了想法才有了創(chuàng)作,你的反饋會(huì)是使我進(jìn)步的最大動(dòng)力。

覺(jué)得還不夠方便?還想要什么功能?告訴我!歡迎反饋,歡迎Star。

我的其他文章推薦:
一句話搞定的Android底部導(dǎo)航欄,一鍵綁定片段,ViewPager
一句話搞定高仿ios底部彈出提示框

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 1.什么是Activity?問(wèn)的不太多,說(shuō)點(diǎn)有深度的 四大組件之一,一般的,一個(gè)用戶交互界面對(duì)應(yīng)一個(gè)activit...
    JoonyLee閱讀 5,855評(píng)論 2 51
  • 一、概述 Activity 作為與用戶交互的一個(gè)窗口,是使用非常頻繁的一個(gè)基本組件。Android系統(tǒng)是通過(guò)Act...
    三也視界閱讀 2,401評(píng)論 3 11
  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 178,812評(píng)論 25 709
  • 融融暖意,晴空萬(wàn)里,怎會(huì)如你在身旁 夏日暴雨,兇猛海嘯,又怎會(huì)明了 后來(lái),陽(yáng)光溫暖,獨(dú)我一人曬衣 酌酒
    青梔__閱讀 274評(píng)論 0 1
  • 一座山-南山花園游記 我喜歡爬山,不是因?yàn)閺男【幼≡谄皆瓫](méi)見(jiàn)過(guò)山,而是天生喜歡挑戰(zhàn),心中總有翻山越嶺的欲望。我于外...
    Real_閱讀 523評(píng)論 2 3

友情鏈接更多精彩內(nèi)容