Android5.0上WebView的android.content.res.Resources$NotFoundException

本文探讨了在Android 5.0系统中创建WebView导致应用崩溃的问题,详细分析了崩溃原因,并提出了解决方案。问题在于WebView资源加载过程中,插件环境下资源缺失,尤其是在使用Chrome内核时。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

        最近遇到一个问题,在android5.0上, 一个插件创建WebView时会Crash,报错:

E/AndroidRuntime(17629): Caused by: android.content.res.Resources$NotFoundException: String resource ID #0x2040002
E/AndroidRuntime(17629):     at android.content.res.Resources.getText(Resources.java:274)
E/AndroidRuntime(17629):     at android.content.res.Resources.getString(Resources.java:360)
E/AndroidRuntime(17629):     at com.android.org.chromium.content.browser.ContentViewCore.setContainerView(ContentViewCore.java:702)
E/AndroidRuntime(17629):     at com.android.org.chromium.content.browser.ContentViewCore.initialize(ContentViewCore.java:608)
E/AndroidRuntime(17629):     at com.android.org.chromium.android_webview.AwContents.createAndInitializeContentViewCore(AwContents.java:617)
E/AndroidRuntime(17629):     at com.android.org.chromium.android_webview.AwContents.setNewAwContents(AwContents.java:756)
E/AndroidRuntime(17629):     at com.android.org.chromium.android_webview.AwContents.<init>(AwContents.java:606)


      但是在主程序中进行创建时,不会有任何问题。

      使用非android5.0系统时,也不会有任何问题。


      查看android5.0上WebView空间的代码,发现在android5.0以后,android系统对WebView控件的实现进行了修改:
     1, 在WebView控件和浏览器内核之间加入了一层代理层。以屏蔽浏览器内核的差异。目前支持WebKit和Chrome两种内核。
     2, 在frameworks/webview目录下新增两个package,一个为webkit,一个为chrome。
     3, 代理层通过配置来加载这两个package中间的一个,对浏览器内核进行选择。
     4, 具体选用哪个浏览器内核,有com.internal.R.string.config_webViewPackageName这个字符串指定的包名来决定。

     在5.0中,com.internal.R.string.config_webViewPackageName的值被获取出来是: com.google.android.webview,也就是chrome内核。在5.1以及以上的版本中,com.internal.R.string.config_webViewPackageName的值被获取出来是: com.android.webview, 也就是webkit内核。

      再看看浏览器资源的加载过程:
      在WebView控件的初始化函数中,会对package的代码和资源进行加载,其中资源部分被加载到了主程序的AssetManager中。
      这里,无论在插件还是在主程序中创建WebView,资源的加载都是在主程序的AssetManager中,这就导致了插件的AssetManager中,并没有浏览器的资源,这就是插件创建WebView找不到资源进而Crash的原因。

      为什么5.1没有问题呢?那是因为5.1上使用的是webkit内核,而webkit没有被Google单独出来成为一个apk,所以加载webkit的资源其实是相当于没有加载资源,所以没有问题。

      后来还发现一个比较有趣的现象,在MIUI系统上,即使是5.1的版本,也是使用的chrome内核,但是这是插件创建WebView却不会出问题,查了一下,发现是因为小米提前把chrome的资源编译到framework中,在创建assertmanager时就进行了加载,使得插件的assertmanager也拥有chrome内核的资源。


       问题搞清楚了之后,这样解决即可:
       在插件创建WebView之前,仿照WebView中加载浏览器内核的代码,先去获取一下com.internal.R.string.config_webViewPackageName的值,获得浏览器内核的包名,然后对插件的assertmanager进行浏览器内核资源的加载。经测试问题解决。
      

       




为啥单元测试的apk编译出来之后会报一下错误,我单元测试编译出来的apk想要单独安装然后使用pm运行,Process crashed before executing the test(s): android.content.res.Resources$NotFoundException: String resource ID #0x0 at android.content.res.Resources.getText(Resources.java:466) at android.content.res.Resources.getString(Resources.java:559) at android.content.Context.getString(Context.java:946) at androidx.startup.AppInitializer.discoverAndInitialize(AppInitializer.java:217) at androidx.startup.AppInitializer.discoverAndInitialize(AppInitializer.java:207) at androidx.startup.InitializationProvider.onCreate(InitializationProvider.java:49) at android.content.ContentProvider.attachInfo(ContentProvider.java:2644) at android.content.ContentProvider.attachInfo(ContentProvider.java:2613) at android.app.ActivityThread.installProvider(ActivityThread.java:8292) at android.app.ActivityThread.installContentProviders(ActivityThread.java:7807) at android.app.ActivityThread.handleBindApplication(ActivityThread.java:7488) at android.app.ActivityThread.-$$Nest$mhandleBindApplication(Unknown Source:0) at android.app.ActivityThread$H.handleMessage(ActivityThread.java:2416) at android.os.Handler.dispatchMessage(Handler.java:107) at android.os.Looper.loopOnce(Looper.java:232) at android.os.Looper.loop(Looper.java:317) at android.app.ActivityThread.main(ActivityThread.java:8705) at java.lang.reflect.Method.invoke(Native Method) at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:580) at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:913) INSTRUMENTATION_RESULT: shortMsg=Process crashed. INSTRUMENTATION_CODE: 0
最新发布
07-31
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值