翻譯Android Weekly--No, You Can Not Override the Home Button... But You Don't Have To!
有關(guān)Home鍵
每個Android開發(fā)者都曾經(jīng)問個這問題,“我能不能覆蓋Home鍵?”。答曰:
當(dāng)然不行!處理返回按鈕已經(jīng)足夠艱難了,如果Home鍵能覆蓋那就更是一團(tuán)糟。
真相如下:
- 如果Home鍵沒有返回桌面,用戶會很失望
- 用戶會困在app中
- 覆蓋不了所有情況
1.由于來電而退出app的情況;
2.掛電話的情況;
3.點擊切換app按鈕的情況。
所以,你很可能是想知道用戶什么時候離開你的應(yīng)用的,而不是把用戶困在應(yīng)用中。如果Application
中有onStop()
方法就相當(dāng)簡單啦,對不對?
Application
中遺失的onStart()
和onStop()
想一想,為什么在onStop()
方法中停止?讓我們深入了解前臺/后臺
(forground/background
)的生命周期,當(dāng)狀態(tài)發(fā)生改變時,能了然于胸。
為什么需要知道當(dāng)前的狀態(tài)呢?假設(shè)我們的應(yīng)用會收到一條通知。如果當(dāng)時應(yīng)用處于前臺運行,肯定會展示出應(yīng)用內(nèi)相應(yīng)的通知界面。相反,如果應(yīng)用處于后臺運行,那在通知欄進(jìn)行展示就比較合適。
例如其他情況,你想知道應(yīng)用從前臺運行到切換至后臺的會話長短,又或者當(dāng)用戶需要處理其他事務(wù)離開時,你需要把你應(yīng)用的緩存清空。
謝天謝地,你可以使用可靠的方式獲取這些信息,而不是使用令人抓狂的ActivityManager.getRunningTask,也不是令人蛋疼的Activity
的生命周期(onStop()
總是不緊跟onStart()
調(diào)用)。
應(yīng)用切換至后臺
從API 14
(Android 4.0 ICS
),我們可以調(diào)用Application.onTrimMemory(int level),這個方法包含了一個等級叫TRIM_MEMORY_UI_HIDDEN,用于記錄應(yīng)用即將進(jìn)入后臺運行。
下面是一個自定義Applicaiton
的使用
public class MyApplication extends Application
{
@Override
public void onTrimMemory(int level) {
super.onTrimMemory(level);
if (level == TRIM_MEMORY_UI_HIDDEN) {
isBackground = true;
notifyBackground();
}
}
}
啊哈!這樣就能知道應(yīng)用切換至后臺運行啦。
手機(jī)熄屏
Application.onTrimMemory(int level)
在手機(jī)熄屏?xí)r不回調(diào)怎么辦?用Intent.ACTION_SCREEN_OFF
注冊BroadcastReceiver
public class MyApplication extends Application {
// ...
@Override
public void onCreate() {
super.onCreate();
// ...
IntentFilter screenOffFilter = new IntentFilter(Intent.ACTION_SCREEN_OFF);
registerReceiver(new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
if (isBackground) {
isBackground = false;
notifyForeground();
}
}
}, screenOffFilter);
}
}
注意:無需監(jiān)聽屏幕點亮的動作,下面會全部搞掂。
應(yīng)用切換至前臺
沒有任何flag
或者trim level
來判斷應(yīng)用切換至前臺,覆寫Activity.onResume()
是最好的方法。在基類Activity
中復(fù)寫它是一個選擇,但無須如此。
一個更簡潔的做法是,利用Application.registerActivityLifeStyleCallbacks(),如名字描述一樣,可以覆寫每一個生命周期函數(shù)。在這個例子中,在不侵入式改動每個Activity
的代碼的前提下,在Activity.onResume()
中執(zhí)行了代碼。
下面是一個自定義Application
:
public class MyApplication extends Application {
// ...
@Override
public void onCreate() {
super.onCreate();
registerActivityLifecycleCallbacks(new Application.ActivityLifecycleCallbacks() {
// ...
@Override
public void onActivityResumed(Activity activity) {
if (isBackground) {
isBackground = false;
notifyForeground();
}
}
// ...
});
}
// ...
}
組織起來
下面是一個應(yīng)用前后臺切換的完整例子。
public class MyApplication extends Application {
// Starts as true in order to be notified on first launch
private boolean isBackground = true;
@Override
public void onCreate() {
super.onCreate();
listenForForeground();
listenForScreenTurningOff();
}
private void listenForForeground() {
registerActivityLifecycleCallbacks(new ActivityLifecycleCallbacks() {
//...
@Override
public void onActivityResumed(Activity activity) {
if (isBackground) {
isBackground = false;
notifyForeground();
}
}
//...
});
}
private void listenForScreenTurningOff() {
IntentFilter screenStateFilter = new IntentFilter(Intent.ACTION_SCREEN_OFF);
registerReceiver(new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
isBackground = true;
notifyBackground();
}
}, screenStateFilter);
}
@Override
public void onTrimMemory(int level) {
super.onTrimMemory(level);
if (level == TRIM_MEMORY_UI_HIDDEN) {
isBackground = true;
notifyBackground();
}
}
private void notifyForeground() {
// This is where you can notify listeners, handle session tracking, etc
}
private void notifyBackground() {
// This is where you can notify listeners, handle session tracking, etc
}
public boolean isBackground() {
return isBackground;
}
}
總結(jié)
- API 14及以上
- 用
Application.onTrimLevel(int level)
和TRIM_MEMORY_UI_HIDDEN
判斷應(yīng)用是否切換至后臺運行。 - 通過
INTENT.ACTION_SCREEN_OFF
注冊廣播接受器監(jiān)聽屏幕熄滅 - 注冊
Activity.registerLifeStyleCallback
監(jiān)聽?wèi)?yīng)用切換至前臺運行 - 別奢望覆寫
Home鍵
- 做個好人