如何修复小米手机特定的RemoteServiceException通知图标问题?

11

我们在安卓6和7的小米手机上遇到了很多崩溃问题:

Fatal Exception: android.app.RemoteServiceException: Bad notification posted from package x.y.z: Couldn't create icon: StatusBarIcon(icon=Icon(typ=RESOURCE pkg=x.y.z id=0x7f0200ad) visible user=0 )
   at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1715)
   at android.os.Handler.dispatchMessage(Handler.java:102)
   at android.os.Looper.loop(Looper.java:163)
   at android.app.ActivityThread.main(ActivityThread.java:6358)
   at java.lang.reflect.Method.invoke(Method.java)
   at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:880)
   at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:770)

我在网上找到了很多类似的崩溃报告和文章。以下是其中的一些:

如何修复:android.app.RemoteServiceException: Bad notification posted from package *: Couldn't create icon: StatusBarIcon

https://medium.com/@Miqubel/the-story-of-a-hard-to-fix-bug-ac6ed819cb49

但是不同的是,我们只在小米手机(Android 6和7)上遇到这些问题,可能不是在更新期间,因为同一用户在相同版本中多次遇到崩溃。有趣的是,我在网上找不到任何关于这个特定情况的信息,我们周围也没有小米手机。我设置通知类似于这样:
NotificationManager notificationManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
    if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.O) {
        int importance = NotificationManager.IMPORTANCE_HIGH;
        NotificationChannel notificationChannel = new NotificationChannel(NOTIFICATION_CHANNEL_ID, NOTIFICATION_CHANNEL_NAME, importance);
        notificationChannel.enableLights(true);
        notificationManager.createNotificationChannel(notificationChannel);
    }
    NotificationCompat.Builder mBuilder = new NotificationCompat.Builder(context, NOTIFICATION_CHANNEL_ID)
            .setSmallIcon(R.drawable.ic_notification)
            .setPriority(NotificationCompat.PRIORITY_MAX)
            .setLargeIcon(BitmapFactory.decodeResource(getResources(), R.mipmap.ic_launcher))
            .setContentText(body == null ? "" : body)
            .setAutoCancel(true)
            .setContentIntent(PendingIntent.getActivity(
                    context,
                    0,
                    pendingIntent,
                    PendingIntent.FLAG_UPDATE_CURRENT
            ));

我们还有Facebook通知,需要以类似的方式设置,但是在不同的Notification类上。我不知道它是否相关。有人遇到过这个问题吗?除了包装setSmallIcon和/或setLargeIcon方法以进行制造商和Android版本检查之外,有什么建议如何修复它?
编辑:我找不到解决方案,但是这里有一些新想法:
- 我们发布了一个新版本,但是排除小米用户的通知并没有帮助!现在我认为问题是由ActivityThread.java中的自定义代码引起的。MIUI可能会在此处触发通知某个事件。在这里,原始Android中有几十个事件,但没有一个事件会触发通知。但是我们的图标有问题,所以它们会崩溃。
- 但是我们的图标有什么问题呢?我们有一个ic_notification,可能不用于此。另一方面,ic_launcher是一个mipmap。也许是这个原因?但是我在小米和mipmap方面找不到任何问题。
- 崩溃报告始终在几个应用程序版本中提到相同的资源ID:0x7f0200ad。这有特殊原因吗?我该如何反向工程我们的应用程序以获取此资源的名称?
编辑2:
- 我使用apktool反向工程了应用程序,但是资源ID不在public.xml中,这似乎是R.java的等效文件。我们的ic_notification和ic_launcher在列表中具有不同的ID。那么这是MIUI找不到的系统资源吗?
编辑3:
- 其他人遇到了同样的问题的第一个证据:

https://xiaomi.eu/community/threads/miui-9.47247/

在波兰论坛上找到了一个临时解决方案:

https://pl.forum.elvenar.com/index.php?threads/problem-z-uruchomieniem-23566.3348/

最后一条评论的翻译是: “我们对于小米手机问题有一个临时解决方案,请尝试在手机设置中禁用来自Elvenar应用程序的强制通知。重启应用程序后,错误应该会消失。”
编辑4: 我们正在使用ShortcutBadger(版本1.1.13)。这里说我们应该为小米徽章使用不同的方法:

https://github.com/leolin310148/ShortcutBadger/wiki/Xiaomi-Device-Support

在1.1.13版本之后,他们取消了对小米的默认支持,您必须使用上面链接中的通知。

还有其他受影响的人使用这个吗?


你是否尝试过使用drawable-anydpi-v21? - 6155031
不,ic_notification只是一个跨越五个密度的简单png。但ic_launcher是一个mipmap。我想知道MIUI能否处理mipmaps?我将在几个小时内更新我的问题,并提供有关此问题的新发现(目前还没有解决方案,情况甚至更糟)。 - Herrbert74
提示:小米有自己的通知系统。 - Martin Zeitler
1
自从2018年12月初以来,我的应用程序在红米设备上也开始频繁崩溃,而在其他设备上则没有出现这种情况。 - KgaboL
"0x7f0200ad。这个有什么特殊的原因吗?我该如何逆向工程我们的应用程序以获取此资源名称?" 按住ctrl键并单击'R',查找id。检查ic_launcher和ic_notification。看看它们是否相匹配。 - Aaditya Brahmbhatt
@AadityaBrahmbhatt 我认为这只给了我本地ID,它们与发布ID不同。而且我们已经根据真正的反向工程建立了上面的内容,该应用程序中没有ID匹配。 - Herrbert74
2个回答

3
我和一个用户遇到了同样的问题。我认为这是由下面这段代码引起的——它是从一个APK反编译出来的,我没有源代码,它来自于shortcutbadger
ResolveInfo.getIconResource()返回了一个无效的资源ID(0x7f0200ad,在所有应用程序中都相同,看起来只在MIUI10上),因此导致了崩溃。
iget-object v1, p0, Lme/leolin/shortcutbadger/impl/XiaomiHomeBadger;->a:Landroid/content/pm/ResolveInfo;

invoke-virtual/range {v1 .. v1}, Landroid/content/pm/ResolveInfo;->getIconResource()I

move-result v1

invoke-virtual {p1, v1}, Landroid/app/Notification$Builder;->setSmallIcon(I)Landroid/app/Notification$Builder;

作为用户,一个简单的解决方法是完全禁用该应用程序的通知 - 至少可以使用它。
作为开发者,可能更容易只使用自己的图标,或在shortcutbadger中添加一些错误处理。
builder.setSmallIcon(R.drawable.myicon);

更新:

现在我明白发生了什么了......在shortcutbadge中有一些奇怪的代码。resolveInfo是默认的主屏幕启动器(MIUI主屏幕)活动,而resolveInfo.getIconResource()则是miui主屏幕的图标。

@TargetApi(Build.VERSION_CODES.JELLY_BEAN)
private void tryNewMiuiBadge(Context context, int badgeCount) throws ShortcutBadgeException {
if (resolveInfo == null) {
    Intent intent = new Intent(Intent.ACTION_MAIN);
    intent.addCategory(Intent.CATEGORY_HOME);
    resolveInfo = context.getPackageManager().resolveActivity(intent, PackageManager.MATCH_DEFAULT_ONLY);
}

if (resolveInfo != null) {
    NotificationManager mNotificationManager = (NotificationManager) context
            .getSystemService(Context.NOTIFICATION_SERVICE);
    Notification.Builder builder = new Notification.Builder(context)
            .setContentTitle("")
            .setContentText("")
            .setSmallIcon(resolveInfo.getIconResource());
    Notification notification = builder.build();

从miuihome.apk反编译后,这里是0x7f0200ad。

<public type="drawable" name="icon_launcher" id="0x7f0200ad" />

那么为什么第三方应用程序会尝试使用MIUI主屏幕的图标来设置通知图标呢?这是为了兼容性而进行的某种黑客攻击,还是只是一个bug?我写了一个简单的应用程序,并使用上述代码片段在模拟器上进行测试,它失败了但没有崩溃应用程序,可能在旧的MIUI上也是同样的情况,因为setSmallIcon(resID)正在寻找来自自己包中具有resID的图标。好消息是,这不是MIUI10的bug,并且只会发生在使用上述代码的应用程序上。


那么,为什么第三方应用程序会尝试设置通知图标为miui home的图标呢?我认为他们这样做是为了让它在小米上运行。真正的问题是为什么(几乎)每个制造商都不必要地进行自定义,只会给我们带来像这样的头痛? - Herrbert74
我现在认为这只是一些有缺陷的代码,从未被捕获,因为它没有任何意义... 在旧版MIUI中,getIconResource()要么返回一个(较低的)值,恰好在大多数应用程序上可用,所以即使它不起作用,也不会崩溃,要么决定继续运行而不是崩溃。 - lex

2
我们购买了一台红米Note 4X。以下是发生的情况:
  • 设备使用MIUI 8.5系统,通知按预期工作,没有崩溃。

  • 我们通过OTA更新升级到MIUI 9.5,打开应用或在某些屏幕之间切换时出现了奇怪的通知。仍然没有崩溃。

  • 我们通过OTA更新升级到MIUI 10.1,先前版本的应用程序在启动时崩溃。当我将ShortcutBadger更新到1.1.22后,崩溃消失了。显然,这是因为小米设备没有在新版库中执行applyCount()方法(它们必须使用applyNotification()),但我没有深入挖掘。

如果有人能详细解释发生了什么,我会保持问题和赏金的开放状态。否则,我们对目前的结果感到满意。针对下一个发布版本,我可能会尝试修复小米的徽章和通知问题,因为它们未正常显示。


我没有使用ShortcutBadger,但今天有一个用户报告了这个问题,你找到根本原因了吗?有什么解决办法吗? - SagiLow
1
@SagiLow 解决方案是更新到最新版本,我没有深入研究。这个库有很多问题,或者说有很多类型的手机和启动器,所以你可能有另一个问题。 - Herrbert74

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接