从 Activity 外部调用 startActivity()?

Posted

技术标签:

【中文标题】从 Activity 外部调用 startActivity()?【英文标题】:Calling startActivity() from outside of an Activity? 【发布时间】:2011-04-11 00:21:11 【问题描述】:

我正在使用AlarmManager 来触发广播信号的意图。以下是我的代码:

AlarmManager mgr = (AlarmManager) getSystemService(Context.ALARM_SERVICE);
Intent i = new Intent(this, Wakeup.class);
try

    PendingIntent pi = PendingIntent.getBroadcast(this, 0, i, 0);
    Long elapsed +=  // sleep time;
    mgr.set(AlarmManager.RTC_WAKEUP, elapsed, pi);

catch(Exception r)

    Log.v(TAG, "RunTimeException: " + r);

我从Activity 调用此代码,所以我不知道我怎么会收到以下错误...

ERROR/androidRuntime(7557): java.lang.RuntimeException: Unable to start receiver com.wcc.Wakeup: android.util.AndroidRuntimeException: Calling startActivity() from outside of an Activity  context requires the FLAG_ACTIVITY_NEW_TASK flag. Is this really what you want?

【问题讨论】:

【参考方案1】:

如果您的 android 版本低于 Android - 6,那么您需要添加此行,否则它将在 Android - 6 以上运行。

...
Intent i = new Intent(this, Wakeup.class);
i.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
...

【讨论】:

是的,这行得通。但是如果你想启动同一个 Activity 的多个实例,这是行不通的。由于 FLAG_ACTIVITY_NEW_TASK 将恢复相同的现有活动 添加标志 FLAG_ACTIVITY_Multiple_TASKS 以解决您的问题 请注意,您在 Android 9+ 上也需要此功能,请参阅 this answer 了解详情【参考方案2】:

对于同一个activity的Multiple Instance,使用下面的sn-p,

注意:这个 sn-p,我在我的 Activity 之外使用。确保您的 AndroidManifest 文件不包含 android:launchMode="singleTop|singleInstance"。如果需要,您可以将其更改为android:launchMode="standard"

Intent i = new Intent().setClass(mActivity.getApplication(), TestUserProfileScreenActivity.class);  
i.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_MULTIPLE_TASK);

// Launch the new activity and add the additional flags to the intent
mActivity.getApplication().startActivity(i);

这对我来说很好。希望,这可以为某人节省时间。如果有人找到更好的方法,请与我们分享。

【讨论】:

请理解 setFlags() 和 addFlag() 方法的区别。您现在所做的就是使用 FLAG_ACTIVITY_NEW_TASK 启动一个活动。就像下面 Cristian 发布的代码一样。 ------------------------------------------ -------------------------i.addFlag(Intent.FLAG_ACTIVITY_MULTIPLE_TASK); i.addFlag(Intent.FLAG_ACTIVITY_NEW_TASK); -------------------------------------------------- ----------------- i.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_MULTIPLE_TASK);【参考方案3】:

Android Doc 说 -

现在强制执行 FLAG_ACTIVITY_NEW_TASK 要求

在 Android 9 中,您无法从非 Activity 启动 Activity 除非您传递意图标志 FLAG_ACTIVITY_NEW_TASK。如果你 尝试在不传递此标志的情况下启动活动,该活动 没有启动,系统会在日志中打印一条消息。

注意:标志要求一直是预期的行为,并且 在低于 Android 7.0(API 级别 24)的版本上强制执行。一个错误 在 Android 7.0 中阻止强制执行标志要求。

这意味着对于(Build.VERSION.SDK_INT <= Build.VERSION_CODES.M) || (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P),必须在从Activity 上下文之外调用startActivity() 时添加Intent.FLAG_ACTIVITY_NEW_TASK

所以最好为所有版本添加标志 -

...
Intent i = new Intent(this, Wakeup.class);
i.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
...

【讨论】:

【参考方案4】:

你没有粘贴你调用startActivity的部分,这是有趣的部分。

您可能在Service 上下文或Application 上下文中调用startActivity

在进行startActivity 调用之前打印“this”以记录cat,看看它指的是什么,有时是不小心使用内部“this”的情况。

【讨论】:

这可能是 OP 遇到的问题。我也遇到过同样的情况。使用 getApplicationContext() 而不是现有 Activity 启动 Activity 将触发此错误。 NICE :) 我很确定这是我整个晚上都试图发现的错误。谢谢,先生。【参考方案5】:

有时在没有显式调用startActivity(...) 的情况下会发生此错误。例如,你们中的一些人可能在 Crashlytics 中看到过这样的堆栈跟踪:

Fatal Exception: android.util.AndroidRuntimeException: Calling startActivity() from outside of an Activity  context requires the FLAG_ACTIVITY_NEW_TASK flag. Is this really what you want?
       at android.app.ContextImpl.startActivity(ContextImpl.java:1597)
       at android.app.ContextImpl.startActivity(ContextImpl.java:1584)
       at android.content.ContextWrapper.startActivity(ContextWrapper.java:337)
       at android.text.style.URLSpan.onClick(URLSpan.java:62)
       at android.text.method.LinkMovementMethod.onTouchEvent(LinkMovementMethod.java:217)
       at android.widget.TextView.onTouchEvent(TextView.java:9522)
       at android.view.View.dispatchTouchEvent(View.java:8968)
       at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2709)
       at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:2425)
       at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2709)
       at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:2425)
       at android.widget.AbsListView.dispatchTouchEvent(AbsListView.java:5303)
       at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2709)
       at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:2425)
       at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2709)
       at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:2425)
       at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2709)
       at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:2425)
       at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2709)
       at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:2425)
       at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2709)
       at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:2425)
       at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2709)
       at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:2425)
       at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2709)
       at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:2425)
       at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2709)
       at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:2425)
       at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2709)
       at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:2425)
       at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2709)
       at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:2425)
       at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2709)
       at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:2425)
       at com.android.internal.policy.impl.PhoneWindow$DecorView.superDispatchTouchEvent(PhoneWindow.java:2559)
       at com.android.internal.policy.impl.PhoneWindow.superDispatchTouchEvent(PhoneWindow.java:1767)
       at android.app.Activity.dispatchTouchEvent(Activity.java:2866)
       at android.support.v7.view.WindowCallbackWrapper.dispatchTouchEvent(WindowCallbackWrapper.java:67)
       at android.support.v7.view.WindowCallbackWrapper.dispatchTouchEvent(WindowCallbackWrapper.java:67)
       at com.android.internal.policy.impl.PhoneWindow$DecorView.dispatchTouchEvent(PhoneWindow.java:2520)
       at android.view.View.dispatchPointerEvent(View.java:9173)
       at android.view.ViewRootImpl$ViewPostImeInputStage.processPointerEvent(ViewRootImpl.java:4706)
       at android.view.ViewRootImpl$ViewPostImeInputStage.onProcess(ViewRootImpl.java:4544)
       at android.view.ViewRootImpl$InputStage.deliver(ViewRootImpl.java:4068)
       at android.view.ViewRootImpl$InputStage.onDeliverToNext(ViewRootImpl.java:4121)
       at android.view.ViewRootImpl$InputStage.forward(ViewRootImpl.java:4087)
       at android.view.ViewRootImpl$AsyncInputStage.forward(ViewRootImpl.java:4201)
       at android.view.ViewRootImpl$InputStage.apply(ViewRootImpl.java:4095)
       at android.view.ViewRootImpl$AsyncInputStage.apply(ViewRootImpl.java:4258)
       at android.view.ViewRootImpl$InputStage.deliver(ViewRootImpl.java:4068)
       at android.view.ViewRootImpl$InputStage.onDeliverToNext(ViewRootImpl.java:4121)
       at android.view.ViewRootImpl$InputStage.forward(ViewRootImpl.java:4087)
       at android.view.ViewRootImpl$InputStage.apply(ViewRootImpl.java:4095)
       at android.view.ViewRootImpl$InputStage.deliver(ViewRootImpl.java:4068)
       at android.view.ViewRootImpl.deliverInputEvent(ViewRootImpl.java:6564)
       at android.view.ViewRootImpl.doProcessInputEvents(ViewRootImpl.java:6454)
       at android.view.ViewRootImpl.enqueueInputEvent(ViewRootImpl.java:6425)
       at android.view.ViewRootImpl$WindowInputEventReceiver.onInputEvent(ViewRootImpl.java:6654)
       at android.view.InputEventReceiver.dispatchInputEvent(InputEventReceiver.java:185)
       at android.os.MessageQueue.nativePollOnce(MessageQueue.java)
       at android.os.MessageQueue.next(MessageQueue.java:143)
       at android.os.Looper.loop(Looper.java:130)
       at android.app.ActivityThread.main(ActivityThread.java:5942)
       at java.lang.reflect.Method.invoke(Method.java)
       at java.lang.reflect.Method.invoke(Method.java:372)
       at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1400)
       at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1195)

你可能想知道你做错了什么,因为跟踪只包括框架代码。好吧,这是一个如何发生这种情况的示例。假设我们在一个片段中。

Activity activity = getActivity();
Context activityContext = activity;
Context appContext = activityContext.getApplicationContext();
LayoutInflater inflater = LayoutInflater.from(appContext); // whoops!
View view = inflater.inflate(R.layout.some_layout, parent, false);
TextView tvWithLinks = (TextView) view.findViewById(R.id.tv_with_links);

tvWithLinks.setMovementMethod(LinkMovementMethod.getInstance()); // whoops!!

现在,当用户单击该文本视图时,您的应用将因上面的堆栈跟踪而崩溃。这是因为 layout inflater 引用了 application context,因此您的 text view 有一个 application context。点击该文本视图隐式调用appContext.startActivity(...)

最后说明:我在 Android 4、5、6 和 7 设备上对此进行了测试。它只影响 4、5 和 6。Android 7 设备显然可以毫无问题地拨打 appContext.startActivity(...)

我希望这对其他人有帮助!

【讨论】:

【参考方案6】:

尝试改成这一行:

PendingIntent pendingIntent = PendingIntent.getBroadcast(getContext(), 0, i, 0);

【讨论】:

【参考方案7】:

我只是想注意,来自活动外部的startActivity 在某些 android 版本(N 和 O-MR1 之间)中是有效的,有趣的是它是 android 源代码中的一个错误!

这是startActivity 实现上面的评论。见here。

从没有 FLAG_ACTIVITY_NEW_TASK 的活动外部调用启动活动是 通常不允许,除非调用者指定了活动应该的任务 ID 被启动。N 和 O-MR1 之间存在一个错误,允许它工作

【讨论】:

解决办法是什么【参考方案8】:

在最新的 Android 设备 9 和 10 中收到通知时,我也遇到了同样的错误。

这取决于您如何处理它的启动模式。使用以下代码:- android:launchMode="singleTask"

使用 Intent 添加此标志:- intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

【讨论】:

【参考方案9】:

当您想在您的应用程序中打开一个活动时,您可以调用 startActivity() 方法,并以 Intent 作为参数。该意图将是您要打开的活动。 首先,您必须创建一个具有该意图的对象,第一个参数是上下文,第二个参数是目标活动类。

Intent intent = new Intent(this, Activity_a.class);
startActivity(intent);

希望这会有所帮助。

【讨论】:

我以审稿人的身份写信,因为您的答案已被标记为低质量。这个答案的缺点是没有解释你的代码做了什么或者为什么它解决了问题中提出的问题。请您添加一些解释性 cmets,记住许多程序员将来可能会阅读您的答案以获得指导。谢谢! 当您想在应用程序中打开一个活动时,您可以使用 Intent 作为参数调用 startActivity() 方法。该意图将是您要打开的活动。首先,您必须创建一个具有该意图的对象,第一个参数是上下文,第二个参数是目标活动类。希望这会有所帮助。 好的。但是,您应该编辑您的答案并将其放在那里,而不是将该信息放在评论中。我很遗憾重复这一点,但对每个答案都这样做很重要。最好的问候。 是的,请说明您的答案为何正确。只有在了解我们为什么要做我们所做的事情时,我们才能学会成为更好的程序员。【参考方案10】:

在我的例子中,我使用了 startActivity 的上下文,在用 ActivityName.this 更改之后。它解决了。我正在使用 util 类中的方法,所以会发生这种情况。

希望对某人有所帮助。

【讨论】:

以上是关于从 Activity 外部调用 startActivity()?的主要内容,如果未能解决你的问题,请参考以下文章

Android中数据的传递

如果 Activity 被销毁,我可以从 Activity 调用静态方法吗?

从适配器调用 Activity 方法

【java安卓】怎么调用另一个安卓activity的对象?

从 Activity 调用 Location 时获取 NullPointerException

如何从 HomeScreen Widget 调用 Activity 中的函数