是否有可能在堆栈上启动一个活动,清除它之前的整个历史?

这种情况

我有一个活动堆栈,要么是A->B->C,要么是B->C(屏幕A选择用户令牌,但许多用户只有一个令牌)。

在屏幕C中,用户可能会采取使屏幕B无效的操作,因此应用程序希望将他们带到屏幕A,而不管它是否已经在堆栈中。屏幕A应该是我的应用程序堆栈上的唯一项。

笔记

还有很多类似的问题,但我还没有找到任何答案。我尝试调用getParent().finish() -这总是导致一个空指针异常。FLAG_ACTIVITY_CLEAR_TOP只在活动已经在堆栈上时才有效。


当前回答

你不应该改变堆栈。Android的后退按钮应该像浏览器一样工作。

我能想出一个办法来做这件事,但这太麻烦了。

使你的活动singleTask添加到AndroidManifest 例子: <活动android: name = " .activities.A " android: label = " @string / A_title” android: launchMode = " singleTask " / > <活动android: name = " .activities.B " android: label = " @string / B_title” android: launchMode = " singleTask " / > 扩展应用程序,它将保存到哪里去的逻辑。

例子:

public class DontHackAndroidLikeThis extends Application {

  private Stack<Activity> classes = new Stack<Activity>();

  public Activity getBackActivity() {
    return classes.pop();
  }

  public void addBackActivity(Activity activity) {
    classes.push(activity);
  }
}

从A到B:

DontHackAndroidLikeThis app = (DontHackAndroidLikeThis) getApplication();
app.addBackActivity(A.class); 
startActivity(this, B.class);

从B到C:

DontHackAndroidLikeThis app = (DontHackAndroidLikeThis) getApplication();
app.addBackActivity(B.class); 
startActivity(this, C.class);

在C:

If ( shouldNotGoBackToB() ) {
  DontHackAndroidLikeThis app = (DontHackAndroidLikeThis) getApplication();
  app.pop();
}

并从堆栈中处理pop()的返回按钮。

再说一次,你不应该这样做:)

其他回答

你不应该改变堆栈。Android的后退按钮应该像浏览器一样工作。

我能想出一个办法来做这件事,但这太麻烦了。

使你的活动singleTask添加到AndroidManifest 例子: <活动android: name = " .activities.A " android: label = " @string / A_title” android: launchMode = " singleTask " / > <活动android: name = " .activities.B " android: label = " @string / B_title” android: launchMode = " singleTask " / > 扩展应用程序,它将保存到哪里去的逻辑。

例子:

public class DontHackAndroidLikeThis extends Application {

  private Stack<Activity> classes = new Stack<Activity>();

  public Activity getBackActivity() {
    return classes.pop();
  }

  public void addBackActivity(Activity activity) {
    classes.push(activity);
  }
}

从A到B:

DontHackAndroidLikeThis app = (DontHackAndroidLikeThis) getApplication();
app.addBackActivity(A.class); 
startActivity(this, B.class);

从B到C:

DontHackAndroidLikeThis app = (DontHackAndroidLikeThis) getApplication();
app.addBackActivity(B.class); 
startActivity(this, C.class);

在C:

If ( shouldNotGoBackToB() ) {
  DontHackAndroidLikeThis app = (DontHackAndroidLikeThis) getApplication();
  app.pop();
}

并从堆栈中处理pop()的返回按钮。

再说一次,你不应该这样做:)

在爪哇:-

 startActivity(new Intent(getApplicationContext(),ChooseServiceActivity.class)
                        .setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TASK));

试试下面的代码,

Intent intent = new Intent(ManageProfileActivity.this, LoginActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP|
                Intent.FLAG_ACTIVITY_CLEAR_TASK| 
                Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent);

试试这个:

Intent logout_intent = new Intent(DashboardActivity.this, LoginActivity.class);
logout_intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
logout_intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
logout_intent.setFlags(Intent.FLAG_ACTIVITY_NO_HISTORY);
startActivity(logout_intent);
finish();

我也在这上面花了几个小时……并且同意FLAG_ACTIVITY_CLEAR_TOP听起来像你想要的:清除整个堆栈,除了正在启动的活动,所以Back按钮退出应用程序。然而,正如Mike Repass所提到的,FLAG_ACTIVITY_CLEAR_TOP只在你正在启动的活动已经在堆栈中时才有效;当活动不存在时,标志不做任何事情。

怎么办呢?将正在启动的活动与FLAG_ACTIVITY_NEW_TASK一起放在堆栈中,这将使该活动成为历史堆栈中新任务的开始。然后添加FLAG_ACTIVITY_CLEAR_TOP标志。

现在,当FLAG_ACTIVITY_CLEAR_TOP在堆栈中查找新活动时,它就在那里,并在清除其他所有内容之前被拉起。

这是我的注销函数;View参数是函数所附加的按钮。

public void onLogoutClick(final View view) {
    Intent i = new Intent(this, Splash.class);
    i.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TASK);
    startActivity(i);
    finish();
}