2012-02-22 39 views
2

我想知道,可能在我的活動的onCreate(),如果應用程序以前按照正常的家庭/後退按鈕關閉,或者如果用戶實際上進入Android設置並選擇強制關閉。是否有可能知道應用程序是否在用戶通過設置強制退出後啓動?

是否可以區分這兩種情況?

編輯:

我想我可能沒有解釋我的意思不夠好。我已經閱讀了活動生命週期文檔,並且我非常瞭解這一點。

基本上,我只想知道什麼時候創建活動,無論用戶是否先前進入了Android設置並強行停止。如果應用程序被強制停止,我想採取一個行動(具體來說,啓動時顯示啓動畫面)。

下面的很多答案都說我可以在onStop()或onDestroy()中放置一個標誌,並且如果用戶強行停止,這些方法將不會被調用。問題是,這些方法將已經被越來越到該點之前調用,因爲這個序列,:

  1. 應用程序處於激活狀態時,使用
  2. 用戶點擊後退按鈕(的onStop()的onDestroy ()調用),或home鍵(的onStop()調用),或最近使用的應用按鈕(的onStop()調用)
  3. 用戶進入Android設置,水龍頭強行停止

在這種情況下,我將有將標誌放入onStop()的共享偏好設置中,但用戶點擊強制停止並且標誌爲stil l在onCreate()中激活。

我不想顯示啓動畫面,除非用戶在設置中強制停止。我知道這不是應該被做的方式......但這個決定不是由我做出的。

+0

總是在onCreate()中顯示啓動畫面。 onCreate()僅在應用程序完成時調用。點擊「返回」按鈕不會調用onDestroy()。退出到主屏幕,然後重新啓動應用程序將調用onResume(),而不是onCreate()。我原來的答案是。你誤解了生命週期。 – 2012-02-22 20:06:50

+0

事實是,當用戶按下後退按鈕時,總是調用onDestroy()。嘗試一下!按回來,然後重新啓動應用程序將導致onCreate()被調用。 – howettl 2012-02-22 20:54:58

+0

此外,onDestroy()/ onCreate()也會在設備方向更改時調用。我絕對不希望我的啓動畫面在任何時候改變方向。 – howettl 2012-02-22 21:02:56

回答

1

我昨天遇到同樣的問題,this是我發現由馬克墨菲說的。 但我設法破解它。

實際上,無論何時按下「強制停止」,系統都會清除應用程序佔用的所有內存,除了2件事情。

1:the shared preference2:sqlite database

所以我通過在應用程序中存儲一個布爾變量來完成這個技巧,操作這個值,然後在應用程序的每次啓動時檢查它並完成它。

CONSTANTS.java

package com.mehuljoisar.forcestopdemo; 

public class CONSTANTS { 

//the changes made to value of below variable will be cleared on force stop,so whenever force stop occurs,the value of variable will be "isForceStopped=true" again. 
    public static boolean isForceStopped = true; 
} 

MainActivity.java

package com.mehuljoisar.forcestopdemo; 

import android.app.Activity; 
import android.content.Context; 
import android.content.Intent; 
import android.os.Bundle; 
import android.view.Menu; 
import android.widget.Toast; 

public class MainActivity extends Activity { 

    private Context mContext; 
    private Intent i; 

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

     initialize(); 


     if(CONSTANTS.isForceStopped) 
     { 
     //this block of code will be executed in 2 scenario, 
     //1: when application is started for very first time 
     //2: when application is started after force stopping 
      Toast.makeText(mContext, "Display Splash-screen and move to next screen", Toast.LENGTH_SHORT).show(); 
     //don't forget this part,it's important to make change so that next time splash screen can be avoided 
      CONSTANTS.isForceStopped=false; 

     //and then launch next screen 
      launchSecondScreen(); 
     } 
     else 
     { 
     //directly launch next screen 
      launchSecondScreen(); 
     } 
    } 

    private void launchSecondScreen() { 
     i.setClass(mContext, SecondActivity.class); 
     startActivity(i); 
     finish(); 
    } 

    private void initialize() { 
     mContext = this; 
     i = new Intent(); 
    } 


    @Override 
    public boolean onCreateOptionsMenu(Menu menu) { 
     // Inflate the menu; this adds items to the action bar if it is present. 
     getMenuInflater().inflate(R.menu.activity_main, menu); 
     return true; 
    } 

} 

SecondActivity.java

package com.mehuljoisar.forcestopdemo; 

import android.app.Activity; 
import android.os.Bundle; 

public class SecondActivity extends Activity{ 

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

} 

AndroidManifest.xml中

<?xml version="1.0" encoding="utf-8"?> 
<manifest xmlns:android="http://schemas.android.com/apk/res/android" 
    package="com.mehuljoisar.forcestopdemo" 
    android:versionCode="1" 
    android:versionName="1.0" > 

    <uses-sdk 
     android:minSdkVersion="8" 
     android:targetSdkVersion="16" /> 

    <application 
     android:allowBackup="true" 
     android:icon="@drawable/ic_launcher" 
     android:label="@string/app_name" 
     android:theme="@style/AppTheme" > 
     <activity 
      android:name="com.mehuljoisar.forcestopdemo.MainActivity" 
      android:label="@string/app_name" > 
      <intent-filter> 
       <action android:name="android.intent.action.MAIN" /> 

       <category android:name="android.intent.category.LAUNCHER" /> 
      </intent-filter> 
     </activity> 
     <activity 
      android:name="com.mehuljoisar.forcestopdemo.SecondActivity" 
      android:label="@string/app_name" > 
      <intent-filter> 
       <action android:name="android.intent.action.VIEW" /> 

       <category android:name="android.intent.category.DEFAULT" /> 
      </intent-filter> 
     </activity> 
    </application> 

</manifest> 

我希望這會有所幫助!

+0

此解決方案無法正常工作 – bCliks 2014-08-03 05:27:15

+0

@bala:是什麼問題?它在我所有的應用程序中都能正常工作。 – 2014-08-03 09:51:10

+0

打開你的應用程序,然後返回到主主屏幕。按住主頁按鈕查看最近使用的應用程序列表。現在從該列表中滑動並關閉您的應用。在這種情況下,該解決方案失敗在這種情況下'isForceStopped = true'。這就是爲什麼我告訴這個解決方案工作不正常。我對麼?任何想法? – bCliks 2014-08-03 13:28:52

0

如果用戶用「強制關閉」關閉了應用程序,則「onStop」方法不會執行。您可以在該方法中放置一個標誌(如保存在文件中,用戶首選項,sql等)。下次應用程序啓動時,請檢查該標誌!

0

覆蓋您的主頁活動中的onBackButtonPressed(在正常情況下,當後退按鈕應用程序主要活動已關閉時)。

在此函數中設置一個變量(比如lastforceClose爲false)SharedPreferences,然後調用onbackbuttonpressed()或finish()的超級方法來完成活動。

下一次,當你開始你可以檢查出在sharedpreferences變量的活動,看看lastforceClose變量,如果它要麼設置爲false或沒有(和你自己現在改爲true)...

你可能也想在這裏處理案例,當用戶有deleted data通過Android設置..(即密鑰不存在於共享首選項xml文件等)

+0

這不會將用戶擊回家而不是返回。它也不考慮用戶回擊,然後進入設置和關閉力量。 – howettl 2012-02-22 18:59:20

+0

當用戶點擊主頁按鈕時,活動並沒有完全被殺死......您可以在最大程度上將其展開,但您永遠不會想要在主頁按鈕點擊(案例1的評論)時停止您的應用程序......此外,你根據正常行爲定義'closed',正如你可以通過覆蓋後退按鈕實際調用'按照正常行爲正確關閉'(第二種情況在註釋中提到的) – 2012-02-22 19:17:48

+0

我知道當用戶點擊主頁按鈕。我不想以編程方式停止應用程序。我只想知道活動何時開始,無論用戶是否在設置中強制停止。 – howettl 2012-02-22 19:21:09

0

我不是100%肯定,但這應該工作。
1)在某些永久存儲上的Activity的onDestroy中保存狀態(正常關閉)。(SharedPreferences或SQLite)
2)在onCreate檢查狀態,如果它是正常關機,那麼應用程序通常關閉,否則強制關閉。重置狀態。

0

我不確定是否有更簡單的方法,但您可以創建一個SharedPreferences對象,該對象使用Activity的onStop函數中的標誌更新。並檢查onCreate中的標誌。如果用戶強制關閉應用程序,我不認爲onStop或onDestroy被調用。

1

首先,它聽起來像你可能會混淆恢復與創建。絕對通讀應用基礎知識,特別是關於Activity Lifecycle的內容。該狀態圖很有幫助。

當用戶「退出」應用程序時,它不一定會被銷燬。實際上,它可能不會被銷燬,除非開發人員在明確銷燬活動的onPause()中加入了一些鉤子。

在Android應用程序的生命週期要了解的一個最重要的概念是這樣的:

Android操作系統完全不會有什麼關於你的應用將持續多久還活着的保證,除了保證它可以在任何時候死亡無論你想不想。

你應該總是相應編寫你的應用程序,它是填充生命週期的每一一個啓動和逐步減少掛鉤很好的做法。當用戶從你的應用程序中退出硬件按鈕時,應用程序不會被銷燬,但當應用程序通過強制退出而關閉時,它將被徹底銷燬。與其試圖想方設法瞭解應用程序是如何關閉的,處理此問題的「正確」方法是將您的onPause()onStop()onDestroy()方法設置爲根據狀態進行適當的清理量並且可能有某種Preference對象或狀態變量,如果onDestroy()被調用,則只會翻轉,因爲從技術上講操作系統仍然可以爲您強制退出應用程序,即使用戶不通過任務經理。

0

您可能會嘗試持續記錄全部應用生命週期事件,然後檢查您的各種onCreate()s記錄的最後一個事件是否正常。如果沒有,你可以合理推斷髮生了什麼事。我猜測,用戶調用的任何關閉力只會導致系統嚮應用發送kill(2),在這種情況下,您將不會收到任何指示。 This可能會有所幫助。

您可能還會考慮是否可以通過其他方式實現您的目標。如果你的應用完全是無狀態的,那麼關閉任何類型的應用都不會有問題。如果您的應用程序具有狀態,則理論上所有的數據/應用程序事務都可以實現,因此您可以避免由於崩潰/強制關閉而導致的不一致。

這是否有幫助還是你在追求別的東西?

0

如何從服務寫入狀態標誌而不是活動?活動生命週期仍將完成,但服務生命週期不應該。

這與您的要求不完全相同,但它可能已經足夠接近。如果您顯示的活動沒有此要求,則您希望停止該服務(並在稍後重新啓動)。或者只是讓服務編寫某種類型的事件流,並使用它來確定啓動時顯示的內容。

相關問題