当应用未运行时,无法从通知调用活动方法



我有一个警报提醒,显示带有额外数据的通知。我想做的是通过意图将额外数据从我的通知发送到我的主活动,然后打开应用程序并在该活动中调用特定方法(位置信息(。该方法将根据我传递locationId打开一个新活动。

当我的应用程序在后台运行时,它工作得很好,但是当它被杀死时,即当应用程序完全关闭时,它会打开应用程序的主要活动,但不调用该方法。

我用于发送意图数据的通知代码:

Intent intent = new Intent(this, MainActivity.class);
intent.putExtra("fromNotification", true);
intent.putExtra("locationId", locationId);
intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP 
| Intent.FLAG_ACTIVITY_SINGLE_TOP | Intent.FLAG_ACTIVITY_NEW_TASK);
PendingIntent pendingIntent = PendingIntent.getActivity(this, 111, intent, 
PendingIntent.FLAG_UPDATE_CURRENT);

这就是我如何称呼我的梅托德:

@Override
protected void onNewIntent(Intent intent) {
super.onNewIntent(intent);
setIntent(intent);
if (getIntent().hasExtra("fromNotification")) {
String locationId = getIntent().getStringExtra("locationId");
CallFetchLocationInfoFromMapsFragment(locationId);
}
}

我也尝试在我的主要活动的onCreate结束时添加它,这应该是解决方案,但它什么也没做:

if (getIntent().hasExtra("fromNotification")) {
String locationId = getIntent().getStringExtra("locationId");
CallFetchLocationInfoFromMapsFragment(locationId);
}

我尝试了更多的解决方案(包括这个,这个,这个,这个,这个,这个,这个,这个,等等(,但无法使其工作。

清单(仅主要活动部分(:

<application
android:name="com.weekendcoder.kemo.hotels.app.AppController"
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity
android:name=".MainActivity"
android:label="@string/app_name"
android:screenOrientation="portrait"
android:theme="@style/AppTheme.NoActionBar"
android:launchMode="singleTask">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
...
</application>

这是我理想工作流程的描述 - 如果我单击通知,它应该打开我的 MainActivity,其中包含一个地图片段。在地图上加载所有内容后,我想从主活动的地图开始一个新活动 - 我想使用从通知中收到的 ID 打开 locationDetails 活动。由于调试在我关闭应用程序后立即停止,因此我无法查明问题所在,但我尝试添加以下内容:

if (getIntent().hasExtra("fromNotification")) 
{
Toast.makeText(MainActivity.this, "Notification!", Toast.LENGTH_LONG).show();
}

在我的开始创建,它确实显示了吐司。我不能在 onCreate 开始时使用此条件,因为地图没有加载和设置,所以我在 onCreate 结束时添加了它。在这种情况下,永远不会调用它。

我可以通过直接从通知中打开新活动来使其工作,但这不是我理想的解决方案。我希望能够在应用程序关闭时从通知调用活动中的任何方法。

我也尝试添加这个:

if (getIntent().hasExtra("fromNotification")) {
String locationId = getIntent().getStringExtra("locationId ");
CallFetchLocationInfoFromMapsFragment(locationId);
Toast.makeText(MainActivity.this, "Notification id: " locationId, Toast.LENGTH_LONG).show();
}else{
Toast.makeText(MainActivity.this, "No notification", Toast.LENGTH_LONG).show();
}

在我的 onCreate 结束时,只是为了看看我是否可以显示此值,但我不断收到"无通知"吐司。

当我在onCreate开始时放置相同的if条件时,它会崩溃:

08-14 18:06:03.319 18591-18591/com.weekendcoder.kemo.hotels D/SQLiteHandler: Fetching notification settings from sqlite db
08-14 18:06:03.329 18591-18591/com.weekendcoder.kemo.hotels W/Settings: Setting airplane_mode_on has moved from android.provider.Settings.System to android.provider.Settings.Global, returning read-only value.
08-14 18:06:03.384 18591-18591/com.weekendcoder.kemo.hotels D/AndroidRuntime: Shutting down VM
08-14 18:06:03.384 18591-18591/com.weekendcoder.kemo.hotels W/dalvikvm: threadid=1: thread exiting with uncaught exception (group=0x41cdc700)
08-14 18:06:03.389 18591-18591/com.weekendcoder.kemo.hotels E/AndroidRuntime: FATAL EXCEPTION: main
java.lang.RuntimeException: Unable to start activity ComponentInfo{com.weekendcoder.kemo.hotels/com.weekendcoder.kemo.hotels.MainActivity}: java.lang.NullPointerException
at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2295)
at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2349)
at android.app.ActivityThread.access$700(ActivityThread.java:159)
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1316)
at android.os.Handler.dispatchMessage(Handler.java:99)
at android.os.Looper.loop(Looper.java:176)
at android.app.ActivityThread.main(ActivityThread.java:5419)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:525)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1046)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:862)
at dalvik.system.NativeStart.main(Native Method)
Caused by: java.lang.NullPointerException
at com.weekendcoder.kemo.hotels.sliderfragments.MapsFragment.fetchlocationDetails(MapsFragment.java:545)
at com.weekendcoder.kemo.hotels.MainActivity.CallFetchlocationDetailsFromMapsFragment(MainActivity.java:661)
at com.weekendcoder.kemo.hotels.MainActivity.onCreate(MainActivity.java:324)
at android.app.Activity.performCreate(Activity.java:5372)
at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1104)
at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2257)
at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2349) 
at android.app.ActivityThread.access$700(ActivityThread.java:159) 
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1316) 
at android.os.Handler.dispatchMessage(Handler.java:99) 
at android.os.Looper.loop(Looper.java:176) 
at android.app.ActivityThread.main(ActivityThread.java:5419) 
at java.lang.reflect.Method.invokeNative(Native Method) 
at java.lang.reflect.Method.invoke(Method.java:525) 
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1046) 
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:862) 
at dalvik.system.NativeStart.main(Native Method) 

编辑:

有一个可能的解决方案是有效的,但我对它不满意,并希望找到更智能/更安全的解决方案。解决方案是在开头获取 id,然后添加以下内容:

if (locationId != null) {
final Handler handler = new Handler();
handler.postDelayed(new Runnable() {
@Override
public void run() {
//Do something after 100ms
CallFetchLocationInfoFromMapsFragment(locationId);
}
}, 5000);
}

这将允许我的应用程序等待 5 秒钟,然后在执行所有其他方法后调用该方法。这有效,但我不确定这是继续这样做的最佳方式,某些设备可能会更慢。最好找到一种方法来检查onCreated是否完全完成执行,然后调用此方法。

在评论(上面(的讨论中,您表示您正在使用launchMode="singleTask"。这是一种特殊的启动模式,对于您的应用程序来说不是必需的。它通常产生的问题多于解决的问题。请删除特殊的启动模式,看看是否有帮助。完成此操作后,在应用未运行时点击通知应启动应用并创建MainActivity的新实例。

您也可能无法在onCreate()中执行您想执行的操作,因为您的Fragment尚未设置。您可能只需要记住您想在onCreate()中做什么(开始位置详细信息(,但要推迟实际执行,直到您设置了Fragment或其他什么onResume()

我认为你需要这个

if (getIntent().hasExtra("fromNotification")) {
String locationId = getIntent().getStringExtra("locationId");
CallFetchLocationInfoFromMapsFragment(locationId);
}

以及在清单中为您的活动设置单个任务启动模式

<activity android:name=".YourActivity"
android:launchMode="singleTask">
...
</activity>

最新更新