首先,我知道不应该真正终止/重新启动 Android 上的应用程序。在我的用例中,我想在服务器向客户端发送一条特定信息的特定情况下对我的应用程序进行出厂重置。
用户只能使用应用程序的一个实例登录服务器(即不允许使用多个设备)。如果另一个实例获得“登录”锁,则该用户的所有其他实例都必须删除其数据(恢复出厂设置),以保持一致性。
强制获取锁是可能的,因为用户可能会删除应用程序并重新安装它,这将导致不同的实例 ID,并且用户将无法再释放锁。因此可以强行获取锁。
由于这种强制可能性,我们需要始终检查具体实例是否具有锁。这是在(几乎)每次向服务器发出请求时完成的。服务器可能会发送“错误的锁 ID”。如果检测到这种情况,客户端应用程序必须删除所有内容。
这就是用例。
我有一个Activity
启动登录的AActivity
L 或应用程序的主Activity
B 取决于sharedPrefs 值。启动 L 或 B 后,它会自行关闭,以便只有 L 或 B 正在运行。因此,在用户已经登录的情况下,B 现在正在运行。
B 开始 C.C 通话startService
为了IntentService
D. 结果是这个堆栈:
(A) > B > C > D
从 D 的 onHandleIntent 方法,一个事件被发送到结果接收者 http://developer.android.com/reference/android/os/ResultReceiver.html R.
R 现在通过向用户提供一个对话框来处理该事件,用户可以在其中选择将应用程序恢复出厂设置(删除数据库、sharedPrefs 等)
恢复出厂设置后,我想重新启动应用程序(以关闭所有活动)并仅再次启动 A,然后启动登录Activity
L 并完成自身:
(A) > L
对话框的 onClick 方法如下所示:
@Override
public void onClick(DialogInterface dialog, int which) {
// Will call onCancelListener
MyApplication.factoryReset(); // (Deletes the database, clears sharedPrefs, etc.)
Intent i = new Intent(MyApp.getContext(), A.class);
i.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
i.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
MyApp.getContext().startActivity(i);
}
这就是MyApp
class:
public class MyApp extends Application {
private static Context context;
@Override
public void onCreate() {
super.onCreate();
context = getApplicationContext();
}
public static Context getContext() {
return context;
}
public static void factoryReset() {
// ...
}
}
问题是如果我使用FLAG_ACTIVITY_NEW_TASK
活动B和C仍在进行。如果我点击登录时的后退按钮Activity
我看到 C,但我想返回主屏幕。
如果我不设置FLAG_ACTIVITY_NEW_TASK
我收到错误:
07-07 12:27:12.272: ERROR/AndroidRuntime(9512): android.util.AndroidRuntimeException: Calling startActivity() from outside of an Activity context requires the FLAG_ACTIVITY_NEW_TASK flag. Is this really what you want?
我无法使用活动Context
,因为ServiceIntent
D 也可能从后台任务中调用,该后台任务由AlarmManager
.
那么我该如何解决这个问题,使活动堆栈变为 (A) > L?