Android P 显示 API 兼容性错误消息

Posted

技术标签:

【中文标题】Android P 显示 API 兼容性错误消息【英文标题】:Android P displaying API compatibility error message 【发布时间】:2018-12-15 01:50:36 【问题描述】:

android P 上运行一个针对 SDK 级别 27 构建的应用程序会有些意外地显示以下对话框(对话框标题是应用程序的名称):

检测到 API 兼容性问题(访问 g.co/dev/appcompat 了解更多信息)

URL 指向this page about restrictions on non-SDK interfaces。我的应用程序本身不使用反射,但它确实使用了 Gson。

Logcat 中没有立即明显的日志消息,除了可能的消息,例如:

访问隐藏字段 Landroid/widget/AbsListView;->mIsChildViewEnabled:Z(浅灰名单,反射)

【问题讨论】:

当您使用 SDK 27 并以 我在实际设备 (Pixel 2016) 上得到了这个,当时运行的是 Android P 的预览版。 感谢 Paul 的及时回复! 【参考方案1】:

在你的应用程序类中的方法 onCreate() init 这个方法可以关闭这个对话框。

private void closeAndroidPDialog()
    try 
        Class aClass = Class.forName("android.content.pm.PackageParser$Package");
        Constructor declaredConstructor = aClass.getDeclaredConstructor(String.class);
        declaredConstructor.setAccessible(true);
     catch (Exception e) 
        e.printStackTrace();
    
    try 
        Class cls = Class.forName("android.app.ActivityThread");
        Method declaredMethod = cls.getDeclaredMethod("currentActivityThread");
        declaredMethod.setAccessible(true);
        Object activityThread = declaredMethod.invoke(null);
        Field mHiddenApiWarningShown = cls.getDeclaredField("mHiddenApiWarningShown");
        mHiddenApiWarningShown.setAccessible(true);
        mHiddenApiWarningShown.setBoolean(activityThread, true);
     catch (Exception e) 
        e.printStackTrace();
    

但这很危险 这种方式只会让你看不到对话框。

【讨论】:

"mHiddenApiWarningShown" 反射工作,但为什么要添加 "android.content.pm.PackageParser$Package" 反射?目的是什么?【参考方案2】:

原来我的一个 Gson 模型暴露了一个返回 File 的 getter。 Gson 使用反射递归检查类的字段,这样做违反了不允许的 SDK 接口的反射。

阅读问题中链接的限制文档让我仔细查看了日志消息,果然,其中一条引起了我的注意:

访问隐藏字段 [...](深灰名单、反射)

我不记得确切的消息,但这里的重点是它在 dark 灰名单中。

我通过针对 SDK 级别 28 并启用新的 StrictMode 功能 detectNonSdkApiUsage() 发现了这一点,我的应用程序将在此情况下崩溃并显示堆栈跟踪:

if (BuildConfig.DEBUG && Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) 
    StrictMode.setVmPolicy(new StrictMode.VmPolicy.Builder()
            .detectNonSdkApiUsage()
            .penaltyLog()
            .build());

堆栈跟踪没有立即洞察力,但它为我指明了正确的方向。

【讨论】:

你把这段代码放在哪里了?在失败的活动中? 在我的例子中,我把它放在了Application的onCreate()中,但是你也可以在Activity的创建中调用它。 很酷,谢谢,我的修复只是针对 android sdk 28,然后将我的构建工具更改为最新的 28.0.2

以上是关于Android P 显示 API 兼容性错误消息的主要内容,如果未能解决你的问题,请参考以下文章

如何在Android Studio中为消息“调用需要API级别21(当前最小值为16)”启用lint错误?

Android P版本(9.0) 新功能介绍和兼容性处理

如果最低 SDK = 14,Lollipop 的 Appcompat、兼容性和支持库

如何从 API 向最终用户显示错误消息?

如何实现winform 可隐藏导航栏

如何根据 React js 中 api 的响应显示成功或错误消息