[譯]玩轉(zhuǎn)Android Application的生命周期(不,不許覆蓋那個Home鍵)

翻譯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ùn)行,肯定會展示出應(yīng)用內(nèi)相應(yīng)的通知界面。相反,如果應(yīng)用處于后臺運(yùn)行,那在通知欄進(jìn)行展示就比較合適。

例如其他情況,你想知道應(yīng)用從前臺運(yùn)行到切換至后臺的會話長短,又或者當(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)入后臺運(yù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)用切換至后臺運(yùn)行啦。

手機(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)用是否切換至后臺運(yùn)行。
  • 通過INTENT.ACTION_SCREEN_OFF注冊廣播接受器監(jiān)聽屏幕熄滅
  • 注冊Activity.registerLifeStyleCallback監(jiān)聽?wèi)?yīng)用切換至前臺運(yùn)行
  • 別奢望覆寫Home鍵
  • 做個好人
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

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

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