Android退出應用最優雅的方式

Hei0615 7年前發布 | 6K 次閱讀 安卓開發 Android開發 移動開發

我們先來看看幾種常見的退出方法(不優雅的方式)

一、容器式

建立一個全局容器,把所有的Activity存儲起來,退出時循環遍歷finish所有Activity

import java.util.ArrayList; 
import java.util.List; 
import android.app.Activity; 
import android.os.Bundle; 
public class BaseActivity extends Activity { 
@Override 
protected void onCreate(Bundle savedInstanceState) { 
super.onCreate(savedInstanceState); 
// 添加Activity到堆棧 
AtyContainer.getInstance().addActivity(this); 
} 
@Override 
protected void onDestroy() { 
super.onDestroy(); 
// 結束Activity&從棧中移除該Activity 
AtyContainer.getInstance().removeActivity(this); 
} 
} 
class AtyContainer { 
private AtyContainer() { 
} 
private static AtyContainer instance = new AtyContainer(); 
private static List activityStack = new ArrayList(); 
public static AtyContainer getInstance() { 
return instance; 
} 
public void addActivity(Activity aty) { 
activityStack.add(aty); 
} 
public void removeActivity(Activity aty) { 
activityStack.remove(aty); 
} 
/** 
* 結束所有Activity 
*/ 
public void finishAllActivity() { 
for (int i = 0, size = activityStack.size(); i if (null != activityStack.get(i)) { 
activityStack.get(i).finish(); 
} 
} 
activityStack.clear(); 
} 
}  

這種方法比較簡單, 但是可以看到activityStack持有這Activity的強引用,也就是說當某個Activity異常退出時,activityStack沒有即使釋放掉引用,就會導致內存問題,接下來我們看一種類似的方式,但是會稍微優雅一點點

二、廣播式

通過在BaseActivity中注冊一個廣播,當退出時發送一個廣播,finish退出

public class BaseActivity extends Activity { 
private static final String EXITACTION = "action.exit"; 
private ExitReceiver exitReceiver = new ExitReceiver(); 
@Override 
protected void onCreate(Bundle savedInstanceState) { 
super.onCreate(savedInstanceState); 
IntentFilter filter = new IntentFilter(); 
filter.addAction(EXITACTION); 
registerReceiver(exitReceiver, filter); 
} 
@Override 
protected void onDestroy() { 
super.onDestroy(); 
unregisterReceiver(exitReceiver); 
} 
class ExitReceiver extends BroadcastReceiver { 
@Override 
public void onReceive(Context context, Intent intent) { 
BaseActivity.this.finish(); 
} 
} 
}  

三、進程式

通過直接殺死當前應用的進程來結束應用,簡單粗暴,而且有(wu)效!

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

這三種都能達到同樣的效果,但是在模擬器上都會彈出 Unfortunately , XXX has stopped 消息提示框,但確實能退出應用。部分真機直接失效,只能finish當前Activity(比如我手上這臺小米note,國產的幾款ROM fw層改動太多,使用這種方式需慎重)

四、 RS優雅式

什么是RS式呢?即Receiver+singleTask 。我們知道Activity有四種加載模式,而singleTask就是其中的一種,使用這個模式之后,當startActivity時,它先會在當前棧中查詢是否存在Activity的實例,如果存在,則將其至于棧頂,并將其之上的所有Activity移除棧。我們打開一個app,首先是一個splash頁面,然后會finish掉splash頁面。跳轉到主頁。然后會在主頁進行N次的跳轉,期間會產生數量不定的Activity,有的被銷毀,有的駐留在棧中,但是棧底永遠是我們的HomeActivity。這樣就讓問題變得簡單很多了。我們只需兩步操作即可優雅的實現app的退出。

1、在HomeActivity注冊一個退出廣播,和第二個廣播式一樣,但是這里只需要在HomeActivity一個頁面注冊即可。

2、設置HomeActivity的啟動模式為singleTask。

當我們需要退出的時候只需要startActivity(this,HomeActivity,class), 再發送一個退出廣播。上面代碼首先會把棧中HomeActivity之上的所有Activity移除出棧,然后接到廣播finish自己。一切OK ! 沒有彈框,不用考慮機型Rom適配。不會有內存問題,就是那么的優雅,簡單!

五、SingleTask改版式

和一些小伙交流之后,很多小伙伴說注冊廣播略顯麻煩,在樓下的小伙伴提出了一種更簡單的方式,思路也很簡單,

1、設置MainActivity的加載模式為singleTask

2、重寫MainActivity中的onNewIntent方法

3、需要退出時在Intent中添加退出的tag

由于很多小伙伴對源碼需求比較熱切,我們這里就直接以代碼的形式為大家講解這種方式

第一步設置MainActivity的加載模式為singleTask

android:launchMode="singleTask" 

第二步重寫onNewIntent()方法

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

第三步 退出

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

六、懶人式

這種方式更加簡單,只需要如下兩步操作

1、將MainActivity設置為singleTask

2、將退出出口放置在MainActivity

我們可以看到很多應用都是雙擊兩次home鍵退出應用,就是基于這樣的方式來實現的,這里在貼一下如何處理連續兩次點擊退出的源碼 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); 
} 

 

來自:http://mobile.51cto.com/android-535181.htm

 

 本文由用戶 Hei0615 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!