
有時(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底部彈出提示框
